IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> C++ 手写简单哈希散列表 -> 正文阅读

[数据结构与算法]C++ 手写简单哈希散列表

仅作笔记使用,准备秋招中,时间不多,来不及解释啦!!!

散列表(Hash table,也叫哈希表),是根据关键码值(Key value)而直接进行访问的数据结构。也就是说,它通过把关键码值映射到表中一个位置来访问记录,以加快查找的速度。这个映射函数叫做散列函数,存放记录的数组叫做散列表

给定表M,存在函数f(key),对任意给定的关键字值key,代入函数后若能得到包含该关键字的记录在表中的地址,则称表M为哈希(Hash)表,函数f(key)为哈希(Hash) 函数。

  • 若关键字为k,则其值存放在f(k)的存储位置上。由此,不需比较便可直接取得所查记录。称这个对应关系f为散列函数,按这个思想建立的表为散列表。

  • 对不同的关键字可能得到同一散列地址,即k1≠k2,而f(k1)==f(k2),这种现象称为冲突(英语:Collision)。具有相同函数值的关键字对该散列函数来说称做同义词。综上所述,根据散列函数f(k)和处理冲突的方法将一组关键字映射到一个有限的连续的地址集(区间)上,并以关键字在地址集中的“像”作为记录在表中的存储位置,这种表便称为散列表,这一映射过程称为散列造表或散列,所得的存储位置称散列地址。

  • 若对于关键字集合中的任一个关键字,经散列函数映象到地址集合中任何一个地址的概率是相等的,则称此类散列函数为均匀散列函数(Uniform Hash function),这就是使关键字经过散列函数得到一个“随机的地址”,从而减少冲突

sample_hashmap.h:

// 创建日期: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; // 桶的初始数量
	int buckets_size; // 桶的数量
	int keys_count; // key的数量
	vector<Node<T>>buckets; // 不定义成指针类型,免去初始化的步骤
	int hashfun(T val); // 哈希函数
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; // 取模是最简单的哈希函数,如果对于string等类型,需要模板偏特化来另外实现
}

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; // 数组元素是结构体对象,.next调出结构体成员.
	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) { // 计算key,并在对应链表中查找有无该数
	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) { // 遍历数组,每个散链表都用while循环删除申请的节点资源
		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() { // 当key的数量比桶的数量多时扩容到原来的两倍
	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;
//}

溜了溜了~~~

  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2022-07-17 16:48:39  更:2022-07-17 16:51:08 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/25 23:45:30-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码