哈希表的实现(哈希捅)

news2025/1/12 0:55:54

今天是哈希表的实现,哈希表也是一种数据结构,我个人认为还是比较简单的,先给大家看看我 的实现代码吧,如下:

#pragma once
#include <iostream>
#include <set>
#include <map>
#include <vector>
#include <string>
#include <assert.h>
//哈希捅实现哈希表
using namespace std;
namespace cc
{
	//哈希捅的每个节点
	template<class K,class V>
	struct hashnode
	{
		pair<K, V> _val;
		hashnode<K, V>* _next = nullptr;
		hashnode(const pair<K,V>& x)
			:_val(x)
		{}
	};
	//哈希表
	template<class K, class V>
	class hash
	{
	public:
		typedef hashnode<K, V> node;
		hash()
		{}
		hash(const hash<K, V>& h)
		{
			_table.resize(h._table.size(), nullptr);
			for (size_t i = 0; i < _table.size(); i++)
			{
				if (h._table[i] != nullptr)
				{
					node* cur = h._table[i];
					while (cur)
					{
						node* copy = new node(cur->_val);
						copy->_next = _table[i];
						_table[i] = copy;
						cur = cur->_next;
					}
				}
			}
			_size = h._size;
		}
		bool insert(const pair<K, V>& x)
		{
			//如果需要扩容或是此时是一个空表
			//注意:这里其实有个负荷因子,因为STL中的用哈希桶实现的哈希表中,负荷因子的大小是1,所以就是相等,如果用线性探测或是二次探测
			//的方法,就不能控制在1,最好控制在0.7-0.8左右就开始扩容,扩容其实根据专业人士的研究,扩容的大小最好是质数的,出现哈希碰撞的几率
			//就比较少
			if (_size == _table.size())
			{
				vector<node*> tem;
				tem.resize(_table.size() == 0 ? 10 : _table.size() * 2, nullptr);
				//旧表节点移动到新表
				for (size_t i = 0; i < _table.size(); i++)
				{
					node* cur = _table[i];
					while (cur)
					{
						cur->_next = tem[i];
						tem[i] = cur;
						cur = cur->_next;
					}
				}
				//移动完成,交换两个表
				tem.swap(_table);
			}
			node* newnode = new node(x);
			int ret = newnode->_val.first % _table.size();
			//头插
			newnode->_next = _table[ret];
			_table[ret] = newnode;
			_size++;
			return true;
		}
		node* find(const K& key)
		{
			int ret = key % _table.size();
			node* prev = nullptr;
			node* cur = _table[ret];
			while (cur)
			{
				if (cur->_val.first == key)
					return cur;
				prev = cur;
				cur = cur->_next;
			}
			return nullptr;
		}
		bool erase(const K& key)
		{
			if (_size == 0)
			{
				cout << "无法删除空表的内容" << endl;
				return false;
			}
			int ret = key % _table.size();
			node* prev = nullptr;
			node* cur = _table[ret];
			while (cur)
			{
				if (cur->_val.first == key)
				{
					if (prev)
						prev->_next = cur->_next;
					else
						_table[ret] = cur->_next;
					delete cur;
					_size--;
					return true;
				}
				prev = cur;
				cur = cur->_next;
			}
			return false;
		}
	private:
		size_t _size = 0;
		vector<node*> _table;
	};
}

上面就是我用哈希捅实现的哈希表,思维逻辑还是比较简单的,但是还是要注意的点。

先来说说哈希表的作用吧。我个人感觉哈希表就是寻找方便,时间复杂度是O(1),个人认为这应该是查找的天花板了吧,就连各种效率都很优的AVL树和红黑树都是log(n),所以个人认为这个数据结构应该是查找的天花板了,但是他的缺点也很明显。我们先来看看下面的图,来看看他的原理:

如上,它的原理其实就是这样的,而哈希捅的实现其实使用链表来实现的,也就是每个桶都挂了一个单向链表,其实不仅可以挂单向链表,双向链表其实也可以挂,但是不用双向链表的原因是,双向链表比单链表的消耗大,所以才用的单链表,而在同一个桶的位置,挂数的时候,我们普遍用头插,这个就不说了,很容易理解,头插的时间复杂度是O(1)。

其实很多人在看到这种情况的时候,可能会懵,这中结构怎么查找的时间复杂度是O(1)呢?如果我查找的桶刚好是一个挂的非常多的,这不是就相当于O(N)了嘛,其实这个我刚开始也有这种疑惑,但是到现在我就比较释然了,我们打个比喻,最坏的情况是所有插入的数在一个桶的位置,这个是最坏的情况,但是不知道大家测试过吗?这种情况除非是人为的,也就是自己故意的,不然基本是不会出现的,因为我测试过,插入随机的十万个左右的树,它的桶数是大概好像是十二万多,而每个桶所挂起的数,最多的才是3个,普遍都是一个桶挂一个数,所以你担心的情况所出现的概率是非常低的。几乎没有,除非你是自己故意的。所以他的每个桶所挂起的数,基本是常数级别的,所以就是O(1),如果你实现是担心,其实他的每个桶的位置不一定要挂链表,也可以挂红黑树啊,这样效率不就是越来越高了嘛,所以我们不要看到一点点的缺点就不放过,而且他的这个缺点出现的概率实在是太低太低了。

我们可以看到的是,它的查找效率几乎是无敌的,因为不管查找什么,他都是映射的关系,直接映射到它的桶的位置,但是其实他的缺点也非常的大,它的缺点就是扩容的消耗太大了,因为他扩容还要把所有的数据再给新扩的这个表拷贝一份,所以这个是他的缺点。

还有就是线性探测与二次探测的方法来实现哈希表,这个后面会陆续的发。

本篇内容如果对你有用的话,希望带你一下赞吧!!!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/999144.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

在 Spring Boot 中配置和使用 JavaMailSender 发送邮件

&#x1f60a; 作者&#xff1a; 一恍过去 &#x1f496; 主页&#xff1a; https://blog.csdn.net/zhuocailing3390 &#x1f38a; 社区&#xff1a; Java技术栈交流 &#x1f389; 主题&#xff1a; 在 Spring Boot 中配置和使用 JavaMailSender 发送邮件 ⏱️ 创作时间…

华为OD机试 - 单词接龙 - 数据结构map、list (Java 2023 B卷 100分)

目录 专栏导读一、题目描述二、输入描述三、输出描述四、输入示例1、输入&#xff1a;2、输出3、说明 五、解题思路1、核心思想&#xff1a;2、核心算法是构建一个map&#xff1a; 六、Java算法源码七、效果展示1、输入2、输出3、说明4、没有移除后再次拼接的情况&#xff0c;改…

40 个 SpringBoot 常用注解让开发加速

一、Spring Web MVC 与 Spring Bean 注解 Spring Web MVC 注解 RequestMapping RequestMapping注解的主要用途是将Web请求与请求处理类中的方法进行映射。Spring MVC和Spring WebFlux都通过RquestMappingHandlerMapping和RequestMappingHndlerAdapter两个类来提供对RequestMa…

盛元广通实验动物中心综合信息管理系统LIMS

动物实验中心需要按照《实验动物管理条例》《实验动物许可证管理办法》等法规要求&#xff0c;对动物从购进、饲养、实验等须建立完整的质量管理体系&#xff0c;本着为实验动物中心减少人为失误,在降低人力成本的同时&#xff0c;保障动物房的安全运行&#xff0c;盛元广通实验…

Git 提交时忽略某些文件

Git 提交时忽略某些文件 ①打开项目目录&#xff0c;找到.gitignore文件 ②编辑.gitignore文件&#xff0c;加上要忽略的文件后缀 以上是针对还没有提交过的文件进行过滤 如果已经这些后缀的文件已经提交过&#xff0c;则需要删除远程的该后缀文件 git rm --cached示例&…

直播美颜工具的工程实现:集成人像美颜sdk的步骤与方法

时下&#xff0c;无论是个人直播、游戏直播还是专业直播&#xff0c;人像美颜技术都可以提高画面质量&#xff0c;吸引更多观众。本文将深入探讨如何实现一个直播美颜工具&#xff0c;着重介绍了集成人像美颜sdk的步骤与方法。 一、选择合适的人像美颜sdk 要实现一个高效的直…

零售商如何提升消费者购物体验?又学一招!

传统的零售方式已经逐渐被数字技术和自动化流程所替代&#xff0c;为消费者提供更为便捷和个性化的购物体验&#xff0c;同时也为零售商提供了更多的机会来提高效率和创造利润。 自动售货机不再局限于传统的零食和饮料&#xff0c;它们能够销售各种商品&#xff0c;从食品到化妆…

容器编排学习(九)服务管理与用户权限管理

一 service管理 1 概述 容器化带来的问题 自动调度&#xff1a;在 Pod 创建之前&#xff0c;用户无法预知 Pod 所在的节点&#xff0c;以及 Pod的IP 地址一个已经存在的 Pod 在运行过程中&#xff0c;如果出现故障&#xff0c;Pod也会在新的节点使用新的IP 进行部署应用程…

Linux基础 - 读取IO信息

背景 尽量不依赖第三方工具 命令 cat /proc/diskstats字段解析 major minor name rio rmerge rsect ruse wio wmerge wsect wuse running use aveqmajor&#xff1a;设备的主设备号。 minor&#xff1a;设备的次设备号。 name&#xff1a;设备名称&#xff0c;如 sda、sdb …

DataX实现Mysql数据同步到ElasticSearch(ES)

Linux环境要求 jdk1.8及以上 python2 准备工作 Linux安装jdk yum install -y java-1.8.0-openjdk.x86_64查看是否安装成功 java -versionlinux安装python yum install -y python查看python版本号&#xff0c;判断是否安装成功 python --version下载DataX&#xff1a; Dat…

前端绘制地铁路线图

前端绘制地铁路线图 前端可以使用多种技术绘制二维地图&#xff0c;以下是几种常见的方法&#xff1a; SVG&#xff1a;SVG是一种基于XML的矢量图形格式&#xff0c;可以使用SVG元素绘制各种形状和路径&#xff0c;包括线、圆、多边形等。可以使用JavaScript库如D3.js来绘制SV…

【胡锡进】大模型量化分析-汇川技术 300124.SZ

接下来&#xff0c;我将使用自回归移动平均模型&#xff08;ARMA&#xff09;、Bollinger带、随机森林回归&#xff08;Random Forest Regression&#xff09;、自回归移动平均法&#xff08;ARIMA&#xff09;和长短期记忆模型&#xff08;LSTM&#xff09;来预测汇川技术未来…

系列四、Nginx的常用命令和配置文件

一、常用命令 1.1、查看nginx的版本号 ./nginx -v 1.2、启动nginx cd /usr/local/nginx/sbin./nginx 1.3、停止nginx cd /usr/local/nginx/sbin./nginx -s stop 1.4、重新加载nginx 说明&#xff1a;该命令用于修改配置文件后&#xff0c;在不重启nginx的情况下使配置文…

iSCSI:提供基于 iSCSI 的网络存储(服务端以及启动器配置)

写在前面 准备考试整理相关笔记博文内容涉及 iSCSI简单介绍&#xff0c;服务端启动器配置以及一个使用Demo理解不足小伙伴帮忙指正 对每个人而言&#xff0c;真正的职责只有一个&#xff1a;找到自我。然后在心中坚守其一生&#xff0c;全心全意&#xff0c;永不停息。所有其它…

html实现邮件模版布局-flex布局table布局-demo

邮件模版布局 flex - 布局简单方便 兼容性差 table - 优点 就是兼容性好&#xff0c;其他没有优点 效果图 flex布局 <!DOCTYPE html> <html lang"en" xmlns:th"http://www.thymeleaf.org"> <head><meta charset"UTF-8"&g…

利用ansbile部署lamp并部署Discuz(非分布式)

目录 一、实验准备 二、设置ansbile的hosts文件 三、在192.168.115.148上完成相关准备 上传Discuz_X3.3_SC_UTF8.zip 配置本地源、确保我们的sr0挂载后可以使用 四、编写roles 创建目录 编写http的main.yml 编写mysql的main.yml 编写phpmain.yml 编写Discuz的main.ym…

调用视频直播点播平台EasyDSS流媒体服务器上传点播文件接口的具体操作步骤

EasyDSS互联网视频云平台可提供一站式的视频转码、点播、直播、推拉流、时移回放等服务&#xff0c;也能支持4K视频的直播、点播等功能。EasyDSS可用于视频点播&#xff0c;并支持OBS、推流相机、EasyRTMP等设备的推流直播&#xff0c;可应用在AR、VR、无人机推流、虚拟直播、教…

JavaScript中迭代:For循环

之前我们学习过if_else这种控制结构&#xff0c;实际上&#xff0c;JavaScript中存在另一中控制结构&#xff0c;那就是循环&#xff0c;本节我们将来学习for循环&#xff1a; ● For循环的基本如下 for (初始值&#xff1b;什么条件下循环会结束&#xff1b;更新初始值&#…

13.Xaml Slider控件 -->滑块控件

1.运行图片 2.运行源码 a.xaml源码 <Grid Name="Grid1"><!--Maximum="100" 最大值Minimum="0" 最小值Value="50" 设定值Orientation

3.k8s dashboard设置域名登录案例(ingress版本为1.3.1)

文章目录 前言一、安装ingress1.1 下载ingress部署文件1.2 查看是否安装成功 二、配置dashboard域名映射2.1.在windows和linux添加上域名映射2.2 生成tls证书2.3 新增ingress配置2.3 验证 总结 前言 前面搭建了集群&#xff0c;配置了账号密码登录&#xff0c;现在配置k8s das…