網站首頁 編程語言 正文
哈希
概念
可以不經過任何比較,直接從表中得到要搜索的元素。 關鍵在于通過某種函數,使元素的存儲位置與它的關鍵碼之間能夠建立 一一映射的關系。這樣就可以通過o(1)的時間復雜度來尋找到元素。
例如數據集合{1,7,4,5,9,6},哈希函數hash(key)=key&capacity?
沖突
hash(7)=7 hash(17)=7,兩個不同的數通過哈希函數映射到了一個位置,產生了沖突。哈希函數設計的越精妙,產生沖突的可能性就越低,但無法避免。
解決方法:
- 閉散列(開放定址法)
- 開散列(拉鏈法)
閉散列
閉散列,(開放定址法)發生沖突時,如果哈希表沒有被填滿,則表內一定還有其他空閑位置,可以把沖突值放到下一個沒有被占用的空余位置上。
如何找到下一個沒有被占用的空位?答:采用線性探測方法。從發生沖突的位置開始,依次向后探測,直到尋找到下一個空位置為止。
線性探測
線性探測的插入
如:在上述的哈希表中插入元素44,由于下標為4的位置放入了元素4,于是從該位置往后++,找到第一個不為空的位置,將44放入。
線性探測的刪除
在尋找要刪除的元素時,依然會根據存放在哈希表的下標開始尋找,比如在上述哈希表中尋找4,在4下標位置直接就可以找到該元素。但如果直接將其刪除,那后續尋找元素44時,就會因為4下標沒有元素,而認為元素44不存在于這張哈希表。所以我們需要設置一個狀態來表示刪除。
哈希表閉散列的模擬實現
我們寫在一個自定義類域 Closehash 里面
準備工作
哈希表中元素狀態
namespace Closehash
{
//哈希表中元素的狀態
enum State
{
EMPTY,
EXIT,
DELETE
};
}
存儲類型用pair即可,但是數據中要包含狀態,我們進行一次封裝
//由于數據需要一個狀態,所以需要將pair<K,V>封裝一層
template<class K,class V>
struct HashDate
{
pair<K, V>_kv;
State _state;
};
開始(畫餅)構建哈希表的內容
template<class K,class V>
class HashTable
{
public:
bool Insert(const pair<K,V>& kv);
HashDate<K, V>* find(const K& key);
bool Erase(const K& key);
private:
vector<HashDate<K,V>> _tables;
size_t _size = 0;
};
閉散列的插入
bool Insert(const pair<K, V>& kv)
{
//if (Find(kv.first)) return false;
//Find實現了再去掉注釋
if (_tables.size() == 0
|| 10 * _size / _tables.size() >= 7)//相當于存了70%
{
//開始擴容
size_t newsize = _tables.size()== 0 ? 10 : _tables.size() * 2;
HashTable<K, V> newHash;
newHash._tables.resize(newsize);
for (auto e: _tables)//注意_tables是HashDate類型 里面有_kv 和_state
{
if (e._state == EXIST)
{
newHash.Insert(e._kv);
}
}
//資本家拷貝方法
_tables.swap(newHash._tables);
}
//走到這里擴容完成 或者空間足夠大
size_t hashi = kv.first % _tables.size();//尋找在表中對應的下標是什么
while (_tables[hashi]._state==EXIST)
{
hashi++;
//走到頭了得回來
hashi%=_tables.size();
}
_tables[hashi]._kv = kv;
_tables[hashi]._state = EXIST;
_size++;
return true;
}
測試用例
void TestHT1()
{
int a[] = { 1, 11, 4, 15, 26, 7, 44 };
HashTable<int, int> ht;
for (auto e : a)
{
ht.Insert(make_pair(e, e));
}
ht.Print();
}
添加個99以驗證擴容功能
閉散列的查找
HashDate<K, V>* Find(const K& key)
{
if (_tables.size() == 0) return nullptr;
size_t hashi = key % _tables.size();
while (_tables[hashi]._state != EMPTY)
{
if (_tables[hashi]._state != DELETE
&& _tables[hashi]._kv.first == key)
{
return &_tables[hashi];
}
hashi++;
hashi% _tables.size();
}
return nullptr;
}
測試用例
cout << ht.Find(4)->_kv.first << endl;
閉散列的刪除
bool Erase(const K& key)
{
HashDate<K,V>* ret = Find(key);
if (ret)
{
ret->_state = DELETE;
--_size;
return true;
}
else
{
return false;
}
}
模擬實現的閉散列中的問題與改進
上述測試用例中使用的是pair<int,int>那我要是用pair<string,int>呢?我的key還可以直接對數組長度取模嗎?
文檔中對這一問題采用了仿函數的解決方法,我們這里也按照該方法模擬一個。
template<class K>
struct HashFunc
{
size_t operator()(const K& key)
{
return (size_t)key;
}
};
// 特化
template<>
struct HashFunc<string>
{
// BKDR
size_t operator()(const string& key)
{
size_t val = 0;
for (auto ch : key)
{
val *= 131;
val += ch;
}
return val;
}
};
template<class K,class V,class Hash=HashFunc<K>>
class HashTable
{
public:
bool Insert(const pair<K,V>& kv);
HashDate<K, V>* find(const K& key);
bool Erase(const K& key);
private:
vector<HashDate<K,V>> _tables;
size_t _size = 0;
};
在每次求 在哈希表中位置的前面添加
Hash hash;
size_t hashi = hash(kv.first) % _tables.size()
測試用例
void TestHT2()
{
string arr[] = { "蘋果", "西瓜", "蘋果", "西瓜", "蘋果", "蘋果", "西瓜", "蘋果", "香蕉", "蘋果", "香蕉" };
//HashTable<string, int, HashFuncString> countHT;
HashTable<string, int> countHT;
for (auto& str : arr)
{
auto ptr = countHT.Find(str);
if (ptr)
{
ptr->_kv.second++;
}
else
{
countHT.Insert(make_pair(str, 1));
}
}
}
測試用例沒加打印...讓我來回看了好幾遍代碼...蠢到無語
原文鏈接:https://blog.csdn.net/qq_68741368/article/details/127752439
相關推薦
- 2022-04-10 小程序 canvas實現圖片預覽,圖片保存
- 2023-06-03 Numpy數值積分的實現_python
- 2022-02-28 react高階函數和函數柯里化 學習
- 2022-12-05 Django中使用AJAX的詳細過程_python
- 2022-11-04 解析Android?Jetpack簡介_Android
- 2023-02-02 redis中的配置以及密碼設置方式_Redis
- 2022-04-21 sql更新語句中update?set?from用法實現_MsSql
- 2022-06-08 記錄一次奇怪的springboot cache redis緩存報錯解決
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支