【算法】- 查找 - 散列表查询(哈希表)

文章目录


前言

散列技术:在记录的存储位置和它的关键字之间建立一个确定的对应关系f,使得每个关键字key对应一个存储位置f(key)
哈希表:采用散列技术将记录存储在一块连续的存储空间中,这块连续存储空间称为散列表或哈希表


一、哈希表的思想

创建哈希表结构体:创建哈希表数据结构HashTable,用来存放散列地址和该散列表存放的数据个数。

cpp 复制代码
//创建哈希表结构
typedef struct HashTable
{
	int count;//记录哈希表中元素个数
	int* elem;//创建哈希表
}HashTable;

初始化哈希表:初始化哈希表主要还是动态开辟存放数据的数组,再初始化count的个数,并将哈希表中的值赋值为空,这里用NULLKEY标志来表示为空。

cpp 复制代码
//初始化哈希表
void InitHashTable(HashTable* HT)
{
	int i;
	m = MAXSIZE;
	//动态开辟内存
	HT->elem = (int*)malloc(m * sizeof(int));
	HT->count = MAXSIZE;
	for (i = 0; i < m; i++)
	{
		HT->elem[i] = NULLKEY;
	}

}

插入哈希表:插入哈希表,通过哈希函数Hash来计算出散列地址,判断该地址是否为NULLKEY,是的话就直接插入,不是的话就将散列地址赋值为下一个地址,继续判断,直到找到了就执行插入操作。

cpp 复制代码
//哈希表的插入
void InsertHashTable(HashTable* HT, int key)
{
	int addr;
	addr = Hash(key);//求散列地址
	while (HT->elem[addr] != NULLKEY)
	{
		addr = (addr + 1) % m;
	}
	HT->elem[addr] = key;
}

查询哈希表:先通过哈希函数获得散列地址,再通过散列地址查询访问。在查询时如果通过散列地址找到的值和key值不等,则散列地址找到下一个操作,直到循环一圈或则通过地址找到的地址为NULLKEY就返回查找失败,成功则返回查找成功。

cpp 复制代码
//哈希表查找
int SearchHash(HashTable HT, int key)
{
	int addr = Hash(key);
	int begin = addr;
	while (HT.elem[addr] != key)
	{
		addr = (addr + 1) % m;
		if (HT.elem[addr] == NULLKEY || addr == begin)
			return 0;
	}
	return 1;
}

二、哈希表

cpp 复制代码
#define MAXSIZE 12
#define NULLKEY -32768
#include <iostream>
using namespace std;

//哈希表查找

//创建哈希表结构
typedef struct HashTable
{
	int count;//记录哈希表中元素个数
	int* elem;//创建哈希表
}HashTable;

int m = 0;

//初始化哈希表
void InitHashTable(HashTable* HT)
{
	int i;
	m = MAXSIZE;
	//动态开辟内存
	HT->elem = (int*)malloc(m * sizeof(int));
	HT->count = MAXSIZE;
	for (i = 0; i < m; i++)
	{
		HT->elem[i] = NULLKEY;
	}

}

//散列函数
int Hash(int key)
{
	return key % m;
}

//哈希表的插入
void InsertHashTable(HashTable* HT, int key)
{
	int addr;
	addr = Hash(key);//求散列地址
	while (HT->elem[addr] != NULLKEY)
	{
		addr = (addr + 1) % m;
	}
	HT->elem[addr] = key;
}

//哈希表查找
int SearchHash(HashTable HT, int key)
{
	int addr = Hash(key);
	int begin = addr;
	while (HT.elem[addr] != key)
	{
		addr = (addr + 1) % m;
		if (HT.elem[addr] == NULLKEY || addr == begin)
			return 0;
	}
	return 1;
}

int main()
{
	int i,result,key;
	HashTable HT;
	int arr[MAXSIZE] = { 12,67,56,16,25,37,22,29,15,47,48,34 };
	//初始化哈希表
	InitHashTable(&HT);
	key = 39;
	for (i = 0; i < m; i++)
	{
		InsertHashTable(&HT, arr[i]);
	}
	result = SearchHash(HT,key);

	if (result)
		printf("查找 %d 成功 \n", key);
	else
		printf("查找 %d 失败。\n", key);

	for (i = 0; i < m; i++)
	{
		key = arr[i];
		result = SearchHash(HT, key);
		if (result)
			printf("查找 %d 成功 \n", key);
		else
			printf("查找 %d 失败。\n", key);
	}
	return 0;
}

总结

散列表查找的效率是最高的,因为它的时间复杂度为O(1)。可惜在实际情况中,会产生冲突(不同的数据在同一地址的情况),但散列表查询还是非常值得的。

相关推荐
明志数科42 分钟前
4D时序标注技术详解:让机器人理解连续动作的数据基础
java·算法·机器人
KaMeidebaby1 小时前
卡梅德生物技术快报|原核表达系统工艺优化:包涵体重折叠 + 分子筛纯化实现功能 RBD 高效制备,附全参数配置
前端·人工智能·算法·数据挖掘·数据分析
无限码力1 小时前
携程0510笔试真题【单数组交换】
算法·携程笔试·携程笔试真题·携程0510笔试真题
zlinear数据采集卡2 小时前
基准电压电路深度解析:从理论参数到ZLinear采集卡的精准参考实战
c语言·单片机·嵌入式硬件·fpga开发·自动化
BlockWay2 小时前
WEEX Labs 周度观察:微软-OpenAI 合作调整与AI 多云趋势
大数据·人工智能·算法·安全·microsoft
风筝在晴天搁浅2 小时前
快手 CodeTop LeetCode 224.基本计算器
数据结构·算法·leetcode
Smoothcloud润云2 小时前
5大功能精修,重构AI算力使用体验!
java·人工智能·windows·算法·重构·编辑器·sublime text
日晨难再3 小时前
C语言&Python&Bash&Tcl:全局变量和局部变量
c语言·python·bash·tcl
计算机安禾3 小时前
【算法分析与设计】第41篇:确定性与非确定性多项式时间:P与NP的形式化
算法
leo__5204 小时前
随机接入退避算法过程模拟实现
网络·算法