散列表(哈希表)的实现

散列函数直接用key%size的形式,size为散列表的大小。

冲突处理采用平方探测法,为保证可以探测到整个散列表空间,散列表大小设置为4k+3形式的素数。

当散列表中的元素过多时会造成性能下降,这时应该倍增散列表的大小,重新计算原来散列表中每个元素在新的散列表中的位置。

散列表的实现

<span style="font-size:18px;">// HashTable.cpp : 定义控制台应用程序的入口点。
//

#include "stdafx.h"
#include<iostream>
using namespace std;

enum state{
	empty,
	deleted,
	busy,
};

template<typename T>
struct HashNode
{
	T key;
	state sta;
	HashNode()
	{
		sta = empty;
	}
};
const int prime_size = 5;
const static int prime[prime_size] //素数表,形式为4k+3,保证可以探测到整个散列表空间
={
	31, 59, 127, 263, 547
};
template<typename T>
class HashTable
{
private:
	HashNode<T>*hashtable;
	double alpha;//当前散列表的充满程度
	int size;//当前散列表的大小
	int current_busy;
	double max_alpha;
	int hash_func(T key)
	{
		return key%size;
	}

	int find_next_prime()
	{
		for (int i = 0; i < prime_size; i++)
		{
			if (prime[i]>size)
				return prime[i];
		}
		return size;
	}

	void reHash()//alpha>max_alpha时调整散列表大小
	{
		int newsize = find_next_prime();
		if (newsize == size)
			return;
		HashNode<T>*newhashtable = new HashNode<T>[newsize];
		for (int i = 0; i < size; i++)
		{
			if (hashtable[i].sta == busy)
			{
				newhashtable[hashtable[i].key % newsize].key = hashtable[i].key;
				newhashtable[hashtable[i].key % newsize].sta = busy;
			}
		}
		delete[]hashtable;
		hashtable = newhashtable;
		size = newsize;
		alpha =get_alpha();
	}
public:

	HashTable()
	{
		max_alpha = 0.80;
		size = prime[0];
		alpha = 0;
		hashtable = new HashNode<T>[size];
	}

	bool insert(T key)
	{
		int count = 0;
		while (true)
		{
			int index = (key + count*count) % size;
			if (hashtable[index].sta != busy)
			{
				hashtable[index].sta = busy;
				hashtable[index].key = key;
				current_busy++;
				alpha = get_alpha();
				if (alpha > max_alpha)
					reHash();
				return true;
			}
			if (hashtable[index].sta == busy&&hashtable[index].key == key)
			{
				cout<<"HashTable中已经有"<<key<<endl;
				return false;
			}
			index = (key - count*count) % size;
			while (index < 0)
			{
				index += size;
			}
			if (hashtable[index].sta != busy)
			{
				hashtable[index].sta = busy;
				hashtable[index].key = key;
				current_busy++;
				alpha = get_alpha();
				if (alpha > max_alpha)
					reHash();
				return true;
			}
			if (hashtable[index].sta == busy&&hashtable[index].key == key)
			{
				cout << "HashTable中已经有" << key << endl;
				return false;
			}
			count++;
		}

	}

	int find(T key)
	{
		int count = 0;
		while (true)
		{
			int index = (key + count*count) % size;
			if (hashtable[index].sta != busy)
			{
				return -1;
			}
			if (hashtable[index].key== key)
			{
				return index;
			}
			index = (key - count*count) % size;
			while (index < 0)
			{
				index += size;
			}
			if (hashtable[index].sta != busy)
			{
				return -1;
			}
			if (hashtable[index].key == key)
			{
				return index;
			}
			count++;
				if (count > size / 2 + 1)
					return -1;
		}
	}

	bool erase(T key)
	{
		int index = find(key);
		if (index >= 0)
		{
			hashtable[index].sta = deleted;
			current_busy--;
			alpha = get_alpha();
			return true;
		}
		return false;
	}

	double get_alpha()
	{
		return double(current_busy) / double(size);
	}

	~HashTable()
	{
		delete[]hashtable;
	}

};

int _tmain(int argc, _TCHAR* argv[])
{
	HashTable<int>aa;
	aa.insert(1);
	aa.insert(2);
	aa.insert(11);
	aa.insert(3);
	aa.insert(5);
	aa.insert(36);
	aa.insert(5);
	cout << aa.find(36) << endl;
	aa.erase(5) ;
	//cout << -7 % 4 << endl;
	system("pause");
	return 0;
}
</span>

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-10-03 11:58:31

散列表(哈希表)的实现的相关文章

散列表(哈希表)

序言: 如果将一系列的记录按照关键字的某种函数存储,那么在查找某个数据的时候就可以直接通过关键字计算出来了,而不在需要“比较”,这样会非常高效,这就是散列技术. 所以散列技术就是:     存储位置=f(关键字)        不管是记录的存储还是查找,都用这种方法 散列技术具有很高的效率,但是使用起来有一些限制.如1个关键字对应多个记录的情况(比如在一个学校的学生中按性别查找,则对应太多的记录),此外散列技术同样不适合于范围查找和排序等操作. 一.散列函数的构造 在设计散了函数的时候主要考虑两

[BS]散列表 哈希表 Hash table

<第五章> 散 列 散列表的实现常常叫做散列(hashing).散列是一种用于以常数平均时间执行插入.删除和查找的技术. 关于散列有一个很重要的概念:散列函数.散列函数是散列的关键处之一,散列函数又是基于映射机制的一种对应关系(一般是多对一的关系). 这章可以分为5个部分:一般想法,散列函数,分离链接法,开放定址法(可分为线性探测.平方探测.双散列).再散列.可扩散列. 本文只写到前四节.即:一般想法,散列函数,分离链接法,开放定址法(可分为线性探测.平方探测.双散列)() 第五章第一节:一般

散列表之直接寻址表

散列表之直接寻址表 直接寻址表的定义 直接寻址表的操作 直接寻址表的代码实现 dataNode的定义 直接寻址表的定义 测试文件 编译运行 总结 注意: 本文中的所有代码你可以在这里: https://github.com/qeesung/algorithm/tree/master/chapter11/11-1/directAddr(这里的会及时更新) 或者这里: http://download.csdn.net/detail/ii1245712564/8793509 找到 散列表之直接寻址表

[数据结构] 散列表(哈希表)

散列表(哈希表) 比较难理解的官方定义:散列表/哈希表(Hash table),是根据关键码值(Key value)而直接进行访问的数据结构.它通过把关键码值映射到表中一个位置来访问记录,以加快查找的速度.这个映射函数叫做散列函数,存放记录的数组叫做散列表. 举个例子,我们在查找中文字典时.假设我们要查找"翁weng",我们根据weng找到了对应的页码233,这个过程就是根据关键码值映射得到了表中的位置.然后我们在字典这个散列表中,根据我们刚才得到的位置 233页,直接访问了"

数据结构与算法----散列/哈希

1. 简介 散列表的实现叫散列hashing,散列用于以常数平均时间执行 插入.删除.查找,不支持排序.findMin.findMax. 查找关键字不需要 比较 在一个记录的存储位置和它的关键字之间建立映射关系:key--f(key)   这个关系就是散列函数/哈希函数.将一些记录存储在一块 连续 的存储空间,这块空间就是散列表/哈希表. 与线性表.树.图比较: 数据元素之间没有什么逻辑关系,也不能用连线图表示出来. 问题: 关键字不同,但通过散列函数计算的结果相同,即出现了冲突 collisi

HashTable-哈希表/散列表

HashTable-散列表/哈希表,是根据关键字(key)而直接访问在内存存储位置的数据结构.它通过一个关键值的函数将所需的数据映射到表中的位置来访问数据,这个映射函数叫做散列函数,存放记录的数组叫做散列表. 构造哈希表的几种方法 直接定址法--取关键字的某个线性函数为散列地址,Hash(Key)= Key 或 Hash(Key)= A*Key + B,A.B为常数. 除留余数法--取关键值被某个不大于散列表长m的数p除后的所得的余数为散列地址.Hash(Key)= Key % P. 平方取中法

哈希表线性探测

HashTable-散列表/哈希表,是根据关键字(key)而直接访问在内存存储位置的数据结构. 它通过一个关键值的函数将所需的数据映射到表中的位置来访问数据,这个映射函数叫做散列函数,存放记录的数组叫做散列表. 哈希冲突/哈希碰撞 不同的Key值经过哈希函数Hash(Key)处理以后可能产生相同的值哈希地址,我们称这种情况为哈希冲突.任意的散列函数都不能避免产生冲突. 我给大家介绍的是哈希表的线性探测,线性探测的基本思路: 1.用一个数据除以散列表的长度,余数是多少,就把这个数放在散列表下标相同

MySQL中哈希表

也称为散列表 由直接寻址表改进而来.先看直接寻址表 当关键字的全域U比较小时,直接寻址是一种简单而有效的技术.加入某应用要用到一个动态集合,其中每个元素都有一个取自全域U={0,1,...,m-1}的关键字.同时假设没有两个元素具有相同的关键字 用一个数组(即直接寻址表)T[0...m-1]表示动态集合,其中每个位置(或称槽或桶)对应全域U中的一个关键字.图5-38说明了这个问题.槽K指向集合的一个关键字为k的元素.如果该集合没有关键字k的元素,则T[k]=NULL 直接寻址存在一个很明显的问题

算法导论-散列表(Hash Table)

目录 引言 直接寻址 散列寻址 散列函数 除法散列 乘法散列 全域散列 完全散列 碰撞处理方法 链表法 开放寻址法 线性探查 二次探查 双重散列 随机散列 再散列问题 完整源码(C++) 参考资料 内容 1.引言 如果想在一个n个元素的列表中,查询元素x是否存在于列表中,首先想到的就是从头到尾遍历一遍列表,逐个进行比较,这种方法效率是Θ(n):当然,如果列表是已经排好序的话,可以采用二分查找算法进行查找,这时效率提升到Θ(logn);  本文中,我们介绍散列表(HashTable),能使查找效率

哈希表 hash table

散列表(Hash table,也叫哈希表),是根据关键码值(Key value)而直接进行访问的数据结构.也就是说,它通过把关键码值映射到表中一个位置来访问记录,以加快查找的速度.这个映射函数叫做散列函数,存放记录的数组叫做散列表. 给定表M,存在函数f(key),对任意给定的关键字值key,代入函数后若能得到包含该关键字的记录在表中的地址,则称表M为哈希(Hash)表,函数f(key)为哈希(Hash) 函数. 首先问题规模确定,例如5台服务器怎么把数据散落在5台上面呢,就用到了hash算法