Species Tree 利用HashTable實現(xiàn)實例代碼
Species Tree 利用HashTable實現(xiàn)
題目再現(xiàn)
題目內(nèi)容:
給定一個物種演化圖, 關(guān)系的表示方式如下: x y : 表示x為y的先祖。 一個物種只會有一個先祖, 一個先祖可以有很多個演化出來的物種, 請你找出每個問題詢問物種的祖父物種(先祖的先祖), 每個物種會使用一個不重復(fù)的編號來表示, 如果該物種沒有祖父物種的話或是不存在, 那么請將他的祖父物種當(dāng)是0。(憑空而生) 保證所有物種間一定有所關(guān)連, 且不會有重復(fù)演化的現(xiàn)象發(fā)生, 即演化圖只會是一棵樹。 輸入格式: 只有一組測資。 第一行會有兩個數(shù)字N、Q,代表總共有N個物種及Q個問題。 接下來N-1行,每一行有兩個數(shù)字x、y, 意義如題目所述。 接下來的Q行,每一行有一個數(shù)字Z, 代表要詢問的物種編號。 測資范圍: 1 < N < 10000 0 < Q < 1000 0 < x, y, z < 1000000 輸出格式: 對于每一個詢問的物種編號,將他們的祖父物種編號加總后再輸出。 輸入樣例: 6 3 1000 2000 1000 3000 2000 4000 3000 5000 5000 6000 5000 6000 1234 輸出樣例: 4000 時間限制:100ms內(nèi)存限制:16000kb
算法實現(xiàn)
1. 簡單數(shù)組下標(biāo)查找法
通過題目的要求,這里可以使用最簡單的方法,因為輸入的x , y中,y的值是確定不變的,所以這里可以定義一個y的取值范圍那么大的數(shù)組,下標(biāo)就是y的值,內(nèi)容就是x的值,這樣查找起來十分方便,時間復(fù)雜度是O(1),但是空間上就會浪費比較多了。
#include <stdio.h>
#include <string.h>
int main(){
int arrBucket[1000000];
int N, Q;
int x, y, z;
long long sum = 0;
memset(arrBucket, 0, sizeof(arrBucket));
scanf("%d %d", &N, &Q);
while(N -- > 1){
scanf("%d %d", &x, &y);
arrBucket[y] = x;
}
while(Q --){
scanf("%d", &z);
if(arrBucket[z] != 0){
if(arrBucket[arrBucket[z]] != 0){
sum += arrBucket[arrBucket[z]];
}
}
}
printf("%lld", sum);
return 0;
}
2. Hash表實現(xiàn)
因為方法1中,浪費空間嚴(yán)重,所以這里使用Hash表實現(xiàn)。
#include <stdio.h>
#include <stdlib.h>
#define NULLKEY -1
typedef struct {
int *elem; //元素
int *elemP; //父元素
int count;
}HashTable;
int Hash(HashTable H, int k){
return k % H.count;
}
void InitHashTable(HashTable *H){
int i;
H->elem = (int *)malloc(sizeof(int) * H->count);
H->elemP = (int *)malloc(sizeof(int) * H->count);
for(i = 0; i < H->count; i ++){
H->elem[i] = NULLKEY;
H->elemP[i] = NULLKEY;
}
}
void InsertHash(HashTable *H, int key, int value){
int addr;
addr = Hash(*H, key);
while(H->elem[addr] != NULLKEY){
addr = (addr + 1) % H->count;
}
H->elem[addr] = key;
H->elemP[addr] = value;
}
int FindHash(HashTable *H, int key, int *addr){
*addr = Hash(*H, key);
while(H->elem[*addr] != key){
*addr = (*addr + 1) % H->count;
if(H->elem[*addr] == NULLKEY || *addr == Hash(*H, key)){
return 0;
}
}
return 1;
}
int main(){
int N, Q;
int x, y, z, addr;
long long sum = 0;
HashTable H;
scanf("%d %d", &N, &Q);
H.count = N - 1;
InitHashTable(&H);
while(N -- > 1){
scanf("%d %d", &x, &y);
InsertHash(&H, y, x);
}
while(Q --){
scanf("%d", &z);
if(FindHash(&H, z, &addr)){
if(FindHash(&H, H.elemP[addr], &addr)){
sum += H.elemP[addr];
}
}
}
printf("%lld", sum);
return 0;
}
3. 用C++的map來實現(xiàn)
看著用C實現(xiàn)起來,相對來說實現(xiàn)的各個功能都要自己寫,這里看一下用C++的STL里面的map來實現(xiàn)上面的題目,非常簡單(不得不說STL真的很好用,但是不如HashTable速度快,空間上也不如HashTable占用的?。?/p>
#include <iostream>
#include <map>
using namespace std;
int main() {
int n, q;
cin >> n >> q;
map<int,int> mp;
map<int,int>::iterator it;
int x, y, z;
for (int i=1; i<n; ++i) {
cin >> x >> y;
mp.insert(pair<int,int>(y,x));
}
int sum = 0;
for (int i=0; i<q; ++i) {
cin >> z;
it = mp.find(z);
if (it != mp.end()) {
it = mp.find(it->second);
if (it != mp.end())
sum += it->second;
}
}
cout << sum;
return 0;
}
感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!
相關(guān)文章
C語言詳細(xì)分析講解關(guān)鍵字const與volatile的用法
在C語言中,我們經(jīng)常會見到const和volatile這兩個關(guān)鍵字,那么我們今天就來介紹下這兩個關(guān)鍵字,提起?const?關(guān)鍵字,我們可能首先想到的是經(jīng)過它修飾的變量便是常量了。其實我們這種想法是錯誤的,其實?const?修飾的變量是只讀的,其本質(zhì)還是變量2022-04-04
C字符串操作函數(shù)實現(xiàn)方法小結(jié)
這篇文章主要介紹了C字符串操作函數(shù)實現(xiàn)方法,實例總結(jié)了C語言字符串操作的相關(guān)技巧,非常具有實用價值,需要的朋友可以參考下2015-04-04
大家注意vector, list, set, map成員函數(shù)erase
set和map是由紅黑樹來實現(xiàn)的,當(dāng)erase的時候迭代器就失效了,也就是說我們要在迭代器失效之前保留一個副本,根據(jù)這個副本我們才能繼續(xù)遍歷下一個元素2013-09-09

