算法思想:

哈希表

什么是哈希表

在前面讨论的各种结构(线性表、树等)中,记录在结构中的相对位置是随机的,和记录的关键字之间不存在确定的关系,因此,在结构中查找记录时需进行一系列和关键字的比较。这一类查找方法建立在“比较”的基础上。

在顺序查找时,比较的结果为“="与“!=”两种可能;

在折半查找、二叉排序树查找和B树查找时,比较的结果为“<"、"="和“>"3种可能。查找的效率依赖于查找过程中所进行的比较次数。

理想的情况是希望不经过任何比较,一次存取便能得到所查记录,那就必须在记录的存储位置和它的关键字之间建立一个确定的对应关系f,使每个关键字和结构中一个惟一的存储位置相对应。因而在查找时,只要根据这个对应关系f找到给定值K的像f(K)。若结构中存在关键字和K相等的记录,则必定在f(K)的存储位置上,由此,不需要进行比较便可直接取得所查记录。在此,我们称这个对应关系f为哈希( Hash)函数,按这个思想建立的表为哈希表。

哈希函数的构造方法

哈希函数是从关键字集合到地址集合的映像。通常,关键字集合比较大,它的元素包括所有可能的关键字,而地址集合的元素仅为哈希表中的地址值。哈希函数其实是一个压缩映像,那么这种情况就不可避免的产生冲突,那么在建造哈希表时不仅要设定一个好的哈希函数,还要设定一种处理冲突的方法。(设定的哈希函数H(key)和处理冲突的方法将一组关键字映像到一个有限的连续的地址集上,并以关键字在地址集中的“像”作为记录在表中的存储位置,这种表就是哈希表,映像的过程为哈希造表或散列,所得的存储位置称哈希地址或散列地址)

(1)直接定址法

取关键字或关键字的某个线性函数值为哈希地址。即H(key)=key 或 H(key)=a*key+b   (a,b为常数)。

举例1:统计1-100岁的人口,其中年龄作为关键字,哈希函数取关键字自身。查找年龄25岁的人口有多少,则直接查表中第25项。

地址 01          02         03   ...           25          26            27  ...   100

年龄 1             2            3   ...            25          26           27   ...   ....

人数 3000    2000  .............             1050

...

举例2:统计解放以后出生人口,其中年份作为关键字,哈希函数取关键字自身加一个常数H(key)=key+(-1948).查找1970年出生的人数,则直接查(1970-1948)=22项即可。

地址 01          02         03   ...           22          23            24  ...

年份 1949     1950      1951   ...     1970

人数                     .............            15000

...

(2)数字分析法

若关键字是以r为基的数(如:以10为基的十进制数),并且哈希表中可能出现的关键字都是事先知道的,则可取关键字的若干数位组成哈希地址。

举例:有80个记录,其关键字为8位十进制数,假设哈希表长,则可取两位十进制数组成哈希地址,为了尽量避免冲突,可先分析关键字。

8 1 3 4 6 5 3 2

8 1 3 7 2 2 4 2

8 1 3 8 7 4 2 2

8 1 3 0 1 3 6 7

8 1 3 2 2 8 1 7

8 1 3 3 8 9 6 7

8 1 3 5 4 1 5 7

8 1 3 6 8 5 3 7

8 1 4 1 9 3 5 5                                          ...........

经分析,发现第一位、第二位都是8,1,第三位只可能取3或4,第八位只可能取2,5或7,所以这四位不可取,那么对于第四、五、六、七位可看成是随机的,因此,可取其中任意两位,或取其中两位与另外两位的叠加求和舍去进位作为哈希地址。

(3)平方取中法

取关键字平方后的中间几位为哈希地址。(较常用的一种)

举例:为BASIC源程序中的标识符键一个哈希表(假设BASIC语言允许的标识符为一个字母或者一个字母和一个数字两种情况,在计算机内可用两位八进制数表示字母和数字),假设表长为512=,则可取关键字平方后的中间9位二进制数为哈希地址。(每3个二进制位可表示1位八进制位,即3个八进制位为9个二进制位)

A :01 (A的ASCII码值为65,65的八进制为101,取后两位表示关键字)

B:02 (B的ASCII码值为66,66的八进制为102,取后两位表示关键字)

...

Z:32(Z的ASCII码值为90,90的八进制为132,取后两位表示关键字)

...

0:60(0的ASCII码值为48,48的八进制为60,取后两位表示关键字)

...

9:71(9的ASCII码值为57,57的八进制为71,取后两位表示关键字)

记录 关键字 关键字的平方 哈希地址(~)

A 0100 0010000 010

I 1100 1210000 210

P1 2061 4310541 310

Q2 2162 4741304 741

(4)折叠法

将关键字分割成位数相同的几部分(最后一部分的位数可不同),然后取这几部分的叠加和(舍去进位)作为哈希地址。适用于关键字位数比较多,且关键字中每一位上数字分布大致均匀时。

举例:根据国际标准图书编号(ISBN)建立一个哈希表。如一个国际标准图书编号  0-442-20586-4的哈希地址为:

5864                                                                                                                5864

4220                                                                                                                0224

+        04                                                                                                             +      04

10088                                                                                                                6092

移位叠加                                                                                                          间接叠加

H(key)=0088(将分割后的每一部分的最低位对齐)                       H(key)=6092(从一端向另一端沿分割界来回叠加)

(5)除留余数法

取关键字被某个不大于哈希表表长m的数p除后所得余数为哈希地址(p为素数)

H(key)=key  MOD  p,p<=m   (最简单,最常用)p的选取很重要

一般情况,p可以选取为质数或者不包含小于20的质因数的合数(合数指自然数中除了能被1和本身整除外,还能被其他数(0除外)整除的数)。

(6)随机数法

选择一个随机函数,取关键字的随机函数值为它的哈希地址。即H(key)=random(key),其中random为随机函数。适用于关键字长度不等时。

总结:实际工作中根据情况不同选用的哈希函数不同,通常,考虑因素如下:

(1)计算哈希函数所需时间(包括硬件指令的因素)

(2)关键字的长度

(3)哈希表的大小

(4)关键字的分布情况

(5)记录的查找频率

常用冲突处理方法:

1.开放定址法:

方法: fi(key)=(f(key)+di) mod m,(di=1,2,3,4...,m−1)fi(key)=(f(key)+di) mod m,(di=1,2,3,4...,m−1)

线性探测:只要一旦发现冲突,就寻找下一个空的散列地址

二次探测:di=12,−12,22,−22,...,q2,−q2di=12,−12,22,−22,...,q2,−q2,目的是不让关键词集中在某块区域,产生堆积

随机探测:didi是一个随机数,但查询时需要设置和插入时相同的随机种子

2.再散列函数法:(再哈希法)

方法:fi(key)=RHi(key) (i=1,2,...k)fi(key)=RHi(key) (i=1,2,...k)

遇到冲突就重新采用一个散列函数计算新的存储位置,可以使关键字不产生聚集

3.链地址法(拉链)

方法:将所有关键字的同义词记录在一个单链表中,在散列表中只存储所有同义词表的头指针

4.建立一个公共溢出区法

方法:为所有冲突的关键字开辟一个公共的溢出区(表)来存放

适用于相对于基本表来说冲突数据很少的情况

实现方法:(哈希表采用数组存储,哈希函数构造和处理冲突的方法是除留余数法+开放定址法)

  1. /****
  2. * Hash Table
  3. *
  4. ****/
  5.  
  6. //#include "Global.h"
  7. #include"stdafx.h"
  8. #include <iostream>
  9. using namespace std;
  10.  
  11. // HashTable Data Structure Definition
  12. // array hashtable
  13. #define tablesize 10
  14. typedef int HashTable[tablesize];
  15. //hash function initialization way
  16. void Initial_HashTable(HashTable &ht)
  17. {
  18. for (int i = ; i < tablesize; i++)
  19. ht[i] = ;
  20. }
  21. //search hashtable function
  22. int Search_HashTable(HashTable &ht,int key)
  23. {
  24. int address = key%tablesize;
  25. int compare = ;
  26. while (compare < tablesize&&ht[address] != key&&ht[address] != )
  27. {
  28. compare++;
  29. address = (address+)%tablesize;
  30. }
  31. if (compare == || ht[address] == )
  32. cout << "can not find elem" << endl;
  33. return address;
  34. }
  35. //insert hashtable function
  36. int Insert_HashTable(HashTable &ht,int key)
  37. {
  38. int res = Search_HashTable(ht,key);
  39. if (ht[res] == )
  40. {
  41. ht[res] = key;
  42. return ;
  43. }
  44. return ;
  45. }
  46. //test function
  47. int main()
  48. {
  49. int data[] = { ,,,,,,, };
  50. HashTable ht;
  51.  
  52. //initialization.
  53. Initial_HashTable(ht);
  54.  
  55. //insert datas.
  56. for (int i = ; i < ; i++)
  57. {
  58. cout << Insert_HashTable(ht, data[i]) << " ";
  59. }
  60. cout << endl;
  61.  
  62. //search.
  63. cout << "25 : " << Search_HashTable(ht, ) << endl;
  64. cout << "35 : " << Search_HashTable(ht, ) << endl;
  65. cout << "145 : " << Search_HashTable(ht, ) << endl;
  66. system("pause");
  67. return ;
  68. }

哈希表查找(散列表查找) c++实现HashMap的更多相关文章

  1. 哈希表(散列表),Hash表漫谈

    1.序 该篇分别讲了散列表的引出.散列函数的设计.处理冲突的方法.并给出一段简单的示例代码. 2.散列表的引出 给定一个关键字集合U={0,1......m-1},总共有不大于m个元素.如果m不是很大 ...

  2. java资料——哈希表(散列表)(转)

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

  3. 哈希表(散列表)—Hash表解决地址冲突 C语言实现

    哈希表(Hash table,也叫散列表),是根据关键码值(Key value)而直接进行访问的数据结构.也就是说,它通过把关键码值映射到表中一个位置来访问记录,以加快查找的速度.具体的介绍网上有很详 ...

  4. 【Python算法】哈希存储、哈希表、散列表原理

    哈希表的定义: 哈希存储的基本思想是以关键字Key为自变量,通过一定的函数关系(散列函数或哈希函数),计算出对应的函数值(哈希地址),以这个值作为数据元素的地址,并将数据元素存入到相应地址的存储单元中 ...

  5. 数据结构---散列表查找(哈希表)概述和简单实现(Java)

    散列表查找定义 散列技术是在记录的存储位置和它的关键字之间建立一个确定的对应关系f,是的每个关键字key对应一个存储位置f(key).查找时,根据这个确定的对应关系找到给定值的key的对应f(key) ...

  6. 算法与数据结构(十二) 散列(哈希)表的创建与查找(Swift版)

    散列表又称为哈希表(Hash Table), 是为了方便查找而生的数据结构.关于散列的表的解释,我想引用维基百科上的解释,如下所示: 散列表(Hash table,也叫哈希表),是根据键(Key)而直 ...

  7. 数据结构(四十二)散列表查找(Hash Table)

    一.散列表查找的基础知识 1.散列表查找的定义 散列技术是在记录的存储位置和它的关键字之间建立一个确定的对应关系f,使得每个关键字key对应一个存储位置f(key).查找时,根据这个确定的对应关系找到 ...

  8. 【PHP数据结构】散列表查找

    上篇文章的查找是不是有意犹未尽的感觉呢?因为我们是真真正正地接触到了时间复杂度的优化.从线性查找的 O(n) 直接优化到了折半查找的 O(logN) ,绝对是一个质的飞跃.但是,我们的折半查找最核心的 ...

  9. 【哈希表】CodeVs1230元素查找

    一.写在前面 哈希表(Hash Table),又称散列表,是一种可以快速处理插入和查询操作的数据结构.哈希表体现着函数映射的思想,它将数据与其存储位置通过某种函数联系起来,其在查询时的高效性也体现在这 ...

随机推荐

  1. XenServer Tools安装

    右键Linux虚拟机,选择 Install XenServer Tools XenCenter 切换到 Console界面 执行如下命令安装: # mount /dev/xvdd /mnt # /mn ...

  2. html中对应Word中的字体和字号

    原文链接:https://blog.csdn.net/xuexizhe88/article/details/79277587:https://blog.csdn.net/qq_31603575/art ...

  3. MATLAB问题小集合

    1.未定义与 'struct' 类型的输入参数相对应的函数 'tb_optparse' 在运行matlab程序时,出现上述错误. 原因是tb_optparse在common文件夹里面,没在robot文 ...

  4. Ajax无法访问回调函数seccess问题

    1,后台返回的数据是标准json格式,前端dataType也是josn, 2,没有跨域访问, 但是一直只执行error方法, 原因出在: 应设置为button按钮,指明类型为button

  5. Mac OS X配置环境变量

    转载注明出处:http://www.jianshu.com/p/7e30b7b7ee48 Mac端环境变量配置 Mac使用bash做为默认的shell MAC OS X环境配置的加载顺序 # 系统级别 ...

  6. React Hook Flow Diagram

    一.概述 Donovon has created this nice flowchart that explains the new lifecycle of a Hooks component. C ...

  7. 学习:CMP/TEST比较指令

    cmp指令:比较 cmp指令,和sub指令的最大的不同点就是影响标志位 不储存结果 1.当前汇编指令为cmp ecx,edx 2.当前ecx寄存器中的地址为00000000,edx寄存器中的地址000 ...

  8. (2)在树莓派安装运行在Python3上的OpenCV

    https://www.jianshu.com/p/56929416b4a1 http://www.eeworld.com.cn/afdz/article_2018030511619.html htt ...

  9. Linux 中 Buffer/Cache内存占用过高解决方法

    在Linux系统中,我们经常用free命令来查看系统内存的使用状态.在一个RHEL6的系统上,free命令的显示内容大概是这样一个状态: 这里的默认显示单位是kb,我的服务器是128G内存,所以数字显 ...

  10. Python高级函数--filter

    def is_palindrome(n): return str(n) == str(n)[::-1] #前两个‘:’表示整个范围,‘-’表示从后面,‘1’表示数据间隔 output = filter ...