網(wǎng)站首頁 編程語言 正文
散列表(Hash table,也叫哈希表),是根據(jù)關(guān)鍵碼值(Key value)而直接進行訪問的數(shù)據(jù)結(jié)構(gòu)。也就是說,它通過把關(guān)鍵碼值映射到表中一個位置來訪問記錄,以加快查找的速度。這個映射函數(shù)叫做散列函數(shù),存放記錄的數(shù)組叫做散列表。
給定表M,存在函數(shù)f(key),對任意給定的關(guān)鍵字值key,代入函數(shù)后若能得到包含該關(guān)鍵字的記錄在表中的地址,則稱表M為哈希(Hash)表,函數(shù)f(key)為哈希(Hash) 函數(shù)。
- 若關(guān)鍵字為k,則其值存放在f(k)的存儲位置上。由此,不需比較便可直接取得所查記錄。稱這個對應(yīng)關(guān)系f為散列函數(shù),按這個思想建立的表為散列表。
- 對不同的關(guān)鍵字可能得到同一散列地址,即k1≠k2,而f(k1)==f(k2),這種現(xiàn)象稱為沖突(英語:Collision)。具有相同函數(shù)值的關(guān)鍵字對該散列函數(shù)來說稱做同義詞。綜上所述,根據(jù)散列函數(shù)f(k)和處理沖突的方法將一組關(guān)鍵字映射到一個有限的連續(xù)的地址集(區(qū)間)上,并以關(guān)鍵字在地址集中的“像”作為記錄在表中的存儲位置,這種表便稱為散列表,這一映射過程稱為散列造表或散列,所得的存儲位置稱散列地址。
- 若對于關(guān)鍵字集合中的任一個關(guān)鍵字,經(jīng)散列函數(shù)映象到地址集合中任何一個地址的概率是相等的,則稱此類散列函數(shù)為均勻散列函數(shù)(Uniform Hash function),這就是使關(guān)鍵字經(jīng)過散列函數(shù)得到一個“隨機的地址”,從而減少沖突
sample_hashmap.h:
// 創(chuàng)建日期:2022-07-13
// 作者:YZM
// 參考:https://github1s.com/ACking-you/my_tiny_stl/blob/HEAD/src/Data_struct_tool/HashTable/sample_HashMap.h
#pragma once
#ifndef SAMPLE_HASHMAP_H
#define SAMPLE_HASHMAP_H
?
#include<iostream>
#include<vector>
using namespace std;
?
template<typename T>
struct Node {
?? ?Node* next;
?? ?T val;
?? ?Node() :next(nullptr), val(0) {};
?? ?Node(T _val) :next(nullptr), val(_val) {};
?? ?Node(T _val, Node* nxt) :next(nxt), val(_val) {};
};
?
template<typename T>
class HashTable {
private:
?? ?const static int init_buckets_size = 49; // 桶的初始數(shù)量
?? ?int buckets_size; // 桶的數(shù)量
?? ?int keys_count; // key的數(shù)量
?? ?vector<Node<T>>buckets; // 不定義成指針類型,免去初始化的步驟
?? ?int hashfun(T val); // 哈希函數(shù)
public:
?? ?HashTable();
?? ?~HashTable();
?? ?int& operator[](int index) const; // 重載[]運算符,哈希表暫時用不到
?? ?void insert(T val); // 插入
?? ?void erase(T val); // 刪除
?? ?bool find(T val); // 尋找
?? ?void expand(); // 擴容
?? ?void clear(); // 清空并釋放資源
?? ?void print(); // 打印檢查
};
#endif?
sample_hashmap.cpp:
#include "sample_hashmap.h"
using namespace std;
template<typename T>
HashTable<T>::HashTable():buckets_size(init_buckets_size), keys_count(0), buckets(vector<Node<T>>(init_buckets_size)){}
template<typename T>
HashTable<T>::~HashTable() {
clear();
}
template<typename T>
int HashTable<T>::hashfun(T val) {
return val % buckets_size;
}
template<typename T>
void HashTable<T>::insert(T val) {
int key = hashfun(val);
Node<T>* newNode = new Node<T>(key);
newNode->next = buckets[key].next;
buckets[key].next = newNode;
++keys_count;
expand();
}
template<typename T>
void HashTable<T>::erase(T val) {
int key = hashfun(val);
Node<T>* cur = buckets[key].next; // 數(shù)組元素是結(jié)構(gòu)體對象,.next調(diào)出結(jié)構(gòu)體成員.
Node<T>* pre = nullptr;
while (cur) {
if (cur->val == val) {
if (pre == nullptr) {
buckets[key].next = cur->next;
delete cur;
}
else {
pre->next = cur->next;
delete cur;
}
return;
}
pre = cur;
cur = cur->next;
}
--keys_count;
}
template<typename T>
bool HashTable<T>::find(T val) {
int key = hashfun(val);
Node<T>* cur = buckets[key].next;
while (cur) {
if (cur->val == val) return true;
cur = cur->next;
}
return false;
}
template<typename T>
void HashTable<T>::clear() {
for (int i = 0; i < buckets_size; ++i) {
Node<T>* cur = buckets[i].next;
while (cur) {
Node<T>* pre = cur;
cur = cur->next;
delete pre;
}
buckets[i].next = nullptr;
}
}
template<typename T>
void HashTable<T>::expand() {
if (keys_count > buckets_size) {
buckets_size <<= 1;
buckets.resize(buckets_size);
}
}
template<typename T>
void HashTable<T>::print() {
for (int i = 0; i < buckets_size; ++i) {
Node<T>* cur = buckets[i].next;
while (cur) {
cout << cur->val << ' ';
cur = cur->next;
}
}
cout << endl;
}
int main() {
HashTable<int>hash;
hash.insert(4);
hash.print();
hash.clear();
hash.print();
hash.insert(4);
hash.print();
hash.erase(4);
hash.print();
return 0;
}
原文鏈接:https://blog.csdn.net/weixin_44178960/article/details/125766207
相關(guān)推薦
- 2022-05-24 C語言的strcpy函數(shù)你了解嗎_C 語言
- 2022-07-04 linux?shell?解析命令行參數(shù)及while?getopts用法小結(jié)_linux shell
- 2022-09-14 iOS實現(xiàn)簡單長截圖_IOS
- 2023-12-15 IDEA去掉activate-power-mode右上角圖標和Power Mode II 進度條10
- 2022-05-08 python如何生成密碼字典_python
- 2022-04-22 頁面跳轉(zhuǎn)之后后退回來頁面隱藏問題。
- 2022-06-25 CentOS?8?安裝調(diào)試KVM的詳細步驟_Kvm
- 2023-01-12 jQuery事件與動畫超詳細講解_jquery
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡單動態(tài)字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支