哈希表,是根據(jù)關(guān)鍵字(Key value)而直接訪問在內(nèi)存存儲位置的數(shù)據(jù)結(jié)構(gòu)。也就是說,它通過計算一個關(guān)于鍵值的函數(shù),將所需查詢的數(shù)據(jù)映射到表中一個位置來訪問記錄,這加快了查找速度。這個映射函數(shù)稱做散列函數(shù),存放記錄的數(shù)組稱做散列表。(摘自維基百科)
為民勤等地區(qū)用戶提供了全套網(wǎng)頁設(shè)計制作服務(wù),及民勤網(wǎng)站建設(shè)行業(yè)解決方案。主營業(yè)務(wù)為成都做網(wǎng)站、成都網(wǎng)站制作、民勤網(wǎng)站設(shè)計,以傳統(tǒng)方式定制建設(shè)網(wǎng)站,并提供域名空間備案等一條龍服務(wù),秉承以專業(yè)、用心的態(tài)度為用戶提供真誠的服務(wù)。我們深信只要達到每一位用戶的要求,就會得到認可,從而選擇與我們長期合作。這樣,我們也可以走得更遠!對不同的關(guān)鍵字可能得到同一散列地址,即k1!=k2,而f(k1)=f(k2),這種現(xiàn)象稱為碰撞(英語:Collision),也叫哈希沖突。
處理哈希沖突的方法有很多種:
閉散列法
開鏈法(哈希桶)
素數(shù)表
字符串哈希算法
在這里我們討論最簡單的閉散列法的線性探測法,學(xué)會了這種方法,就可以在線性探測法的思想基礎(chǔ)上領(lǐng)會其他方法。
線性探測法
定義:通過散列函數(shù)hash(key),找到關(guān)鍵字key在線性序列中的位置,如果當(dāng)前位置已經(jīng)有了一個關(guān)鍵字,就長生了哈希沖突,就往后探測i個位置(i小于線性序列的大?。钡疆?dāng)前位置沒有關(guān)鍵字存在。
#pragma once #include#include using namespace std; enum State { EMPTY, EXIST, DELETE }; template struct DefaultFunc { size_t operator()(const T& data) { return (size_t)data; } }; struct StringFunc { size_t operator()(const string& str) { size_t sum = 0; for (size_t i = 0; i < str.size(); ++i) { sum += str[i]; } return sum; } }; template > class HashTable { public: HashTable(); HashTable(const size_t size); bool Push(const K& data);//增 bool Remove(const K& data);//刪 size_t Find(const K& data);//查 bool Alter(const K& data, const K& newdata);//改 void Print();//打印哈希表 protected: size_t HashFunc(const K& data);//散列函數(shù)(哈希函數(shù)) void Swap(HashTable & x); protected: K* _table;//哈希表 State* _state;//狀態(tài)表 size_t _size; size_t _capacity; FuncModel _HF;//區(qū)分默認類型的哈希函數(shù)和string類型的哈希函數(shù) };
.cpp文件
#define _CRT_SECURE_NO_WARNINGS 1 #include"HashTable.h" template> HashTable ::HashTable() :_table(NULL) , _state(NULL) , _size(0) , _capacity(0) {} template > HashTable ::HashTable(const size_t size) :_table(new K[size]) , _state(new State[size]) , _size(0) , _capacity(size) { //這里別用memset()來初始化_state,對于枚舉類型的動態(tài)內(nèi)存不能用memset初始化 //老老實實一個一個初始化 for (size_t i = 0; i < _capacity; i++) { _state[i] = EMPTY; } } template > size_t HashTable ::HashFunc(const K& data) { return _HF(data)%_capacity;//Mod哈希表的容量,找到在哈希表中的位置, //其實在這里最好Mod一個素數(shù) } template > void HashTable ::Swap(HashTable & x)//交換兩個哈希表 { swap(_table, x._table); swap(_state, x._state); swap(_size, x._size); swap(_capacity, x._capacity); } template > bool HashTable ::Push(const K& data) { if if (_size *10 >= _capacity* 8)//載荷因子不超過0.8 { HashTable tmp(2 * _capacity + 2); for (size_t i = 0; i < _capacity; ++i) { if (_state[i] == EXIST) { size_t index = HashFunc(_table[i]); while (tmp._state[index] == EXIST) { index++; } tmp._table[index] = _table[i]; tmp._state[index] = EXIST; } } Swap(tmp); } size_t index = HashFunc(data); while (_state[index] == EXIST) { index++; } _table[index] = data; _state[index] = EXIST; _size++; return true; } template > void HashTable ::Print() { for (size_t i = 0; i < _capacity; ++i) { if (_state[i] == EXIST) { printf("_table[%d]:", i); cout << _table[i] << "->存在"; } else if (_state[i] == DELETE) { printf("_table[%d]:", i); cout << _table[i] << "->刪除"; } else { printf("_table[%d]:空", i); } cout << endl; } } template > bool HashTable ::Remove(const K& data) { if (_size > 0) { size_t index = Find(data); if (index > 0) { _state[index] = DELETE; _size--; return true; } else return false; } return false; } template > size_t HashTable ::Find(const K& data) { size_t index = HashFunc(data); size_t time = _capacity; while (time--) { if (_table[index++] == data) { return --index; } if (index == _capacity) { index = 0; } } return -1; } template > bool HashTable ::Alter(const K& data, const K& newdata) { size_t index = Find(data); if (index > 0) { _state[index] = DELETE; if (Push(newdata)) return true; else return false; } return false; }
在實現(xiàn)過程中要注意的問題有以下幾點:
對于線性探測來說,有時候會遇到一開始探測的位置就在哈希table的最后的部分,但是因為哈希沖突key值被沖突到了哈希table的最前部分,所以探測到了table尾后將index置為0,簡單又粗暴。
對于對哈希表中的數(shù)據(jù)的刪除是屬于弱刪除,也就是說刪除并沒有刪除數(shù)據(jù),只是把數(shù)據(jù)的狀態(tài)_state置為DELETE。
當(dāng)載荷因子超過0.8時就得增容,載荷因子越高哈希沖突越多,不命中率越高。CPU緩存會大大升高。載荷因子a=填入表中元素的個數(shù)/散列表長度。
對代碼的兩點說明:
在這里我將模板聲明與定義分開,涉及了模板的分離編譯,對模板分離編譯還不太清楚的可以查看博主博客http://helloleex.blog.51cto.com/10728491/1769994
并且為了增強代碼的復(fù)用性,我使用了仿函數(shù)來區(qū)別調(diào)用默認類型(基本類型,自定義類型)和string類型,使調(diào)用更加靈活
另外有需要云服務(wù)器可以了解下創(chuàng)新互聯(lián)scvps.cn,海內(nèi)外云服務(wù)器15元起步,三天無理由+7*72小時售后在線,公司持有idc許可證,提供“云服務(wù)器、裸金屬服務(wù)器、高防服務(wù)器、香港服務(wù)器、美國服務(wù)器、虛擬主機、免備案服務(wù)器”等云主機租用服務(wù)以及企業(yè)上云的綜合解決方案,具有“安全穩(wěn)定、簡單易用、服務(wù)可用性高、性價比高”等特點與優(yōu)勢,專為企業(yè)上云打造定制,能夠滿足用戶豐富、多元化的應(yīng)用場景需求。