哈希查找

╰+哭是因爲堅強的太久メ 2022-01-23 09:34 352阅读 0赞

提起哈希,Java中的Hashtable类,它是由 key/value 的键值对组成的集合,它就是应用了哈希技术。

那什么是哈希查找呢?在弄清楚什么是哈希查找之前,我们要弄清楚哈希技术,哈希技术是在记录的存储位置和记录的 key 之间建立一个确定的映射 f(),使得每个 key 对应一个存储位置 f(key)。若查找集合中存在这个记录,则必定在 f(key) 的位置上。哈希技术既是一种存储方法,也是一种查找方法。

六种哈希函数 f(key) 的构造方法:

1、直接定址法

哈希地址:f(key) = a*key+b (a,b为常数)

这种方法的优点是:简单,均匀,不会产生冲突。但是需要事先知道 key 的分布情况,适合查找表较小并且连续的情况。

2、数字分析法

比如我们的11位手机号码“136xxxx5889”,其中前三位是接入号,一般对应不同运营公司的子品牌,如130是联通如意通,136是移动神州行等等。中间四位表示归属地。最后四位才是用户号。

若我们现在要存储某家公司员工登记表,如果用手机号码作为 key,那么极有可能前7位都是相同的,所以我们选择最后四位作为 f(key) 就是不错的选择。

3、平方取中法

故名思义,比如 key 是1234,那么它的平方就是1522756,再抽取中间的3位就是227作为 f(key) 。

4、折叠法

折叠法是将 key 从左到右分割成位数相等的几个部分(最后一部分位数不够可以短些),然后将这几部分叠加求和,并按哈希表的表长,取后几位作为 f(key) 。

比如我们的 key 是 9876543210,哈希表的表长为3位,我们将 key 分为4组,987|654|321|0 ,然后将它们叠加求和 987+654+321+0=1962,再取后3位即得到 f(key) = 962 。

5、除留余数法

哈希地址:f(key) = key mod p (p<=m) m为哈希表表长。

这种方法是最常用的哈希函数构造方法。下面的代码中也使用了这种方法。

6、随机数法

哈希地址:f(key) = random(key)

这里 random 是随机函数,当 key 的长度不等时,采用这种方法比较合适。

哈希函数冲突的两种解决方法:

我们设计得再好的哈希函数也不可能完全避免冲突,当我们使用哈希函数后发现有 key1 != key2,但却有 f(key1) = f(key2) ,即发生冲突。

1、开放定址法:

开放定址法就是一旦发生了冲突,就去寻找下一个空的哈希地址,只要哈希表足够大,空的哈希地址总是能找到,然后将记录插入。这种方法是最常用的解决冲突的方法。下面的代码中也使用了这种方法。

代码:

HashSearch.java

  1. public class Demo1 {
  2. //初始化哈希表
  3. public static int[] hashTable = new int[8];
  4. //原始数据
  5. public static int[] list = new int[] {13, 29, 27, 28, 26, 30, 38, 35};
  6. public static void main(String[] args) {
  7. //待查找数据
  8. int data = 35;
  9. for(int i=0;i<hashTable.length;i++) {
  10. insert(hashTable, list[i]);
  11. }
  12. //遍历哈希列表
  13. System.out.println("哈希列表:"+showHashTable(hashTable));
  14. //查询数据在哈希中下标
  15. int result = search(hashTable,data);
  16. //判断查找是否成功,成功则返回下标
  17. if(result == -1) {
  18. System.out.println("查找失败");
  19. }else {
  20. System.out.println("数据位置为:"+result);
  21. }
  22. }
  23. /**
  24. * 哈希查找
  25. * @param hashTable
  26. * @param data
  27. * @return
  28. */
  29. public static int search(int[] hashTable,int data) {
  30. //除留余数法
  31. int model = data % hashTable.length;
  32. int hashAddress = model;
  33. while(hashTable[hashAddress] != data) {
  34. // 利用 开放定址法 解决冲突
  35. hashAddress = (++hashAddress) % hashTable.length;
  36. //查找到开放单元(值为0代表未开放单元)或者下标循环回到原点,查找失败
  37. if(hashTable[hashAddress] == 0 || hashAddress == model) {
  38. return -1;
  39. }
  40. }
  41. //查找成功返回下标
  42. return hashAddress;
  43. }
  44. /**
  45. * @method 哈希插入
  46. * @param hashTable
  47. * @param data
  48. */
  49. public static void insert(int[] hashTable,int data) {
  50. //除留余数法
  51. int hashAddress = data % hashTable.length;
  52. //int数组初始为0,判断当前下标数值是否0,不为0说明发送冲突,则下标移到下一位
  53. while(hashTable[hashAddress] != 0) {
  54. // 利用 开放定址法 解决冲突
  55. hashAddress = (++hashAddress) % hashTable.length;
  56. }
  57. //将当前值存入字典中
  58. hashTable[hashAddress] = data;
  59. }
  60. //遍历哈希列表
  61. public static String showHashTable(int[] hashTable) {
  62. StringBuilder sb = new StringBuilder();
  63. for(int i:hashTable) {
  64. sb.append(i+" ");
  65. }
  66. return sb.toString();
  67. }
  68. }运行结果:

哈希列表:38 35 26 27 28 13 29 30
数据位置为:1

发表评论

表情:
评论列表 (有 0 条评论,352人围观)

还没有评论,来说两句吧...

相关阅读

    相关 查找

    哈希查找是一种通过设计所存储数据元素 与其存放地址之间的映射关系(函数关系)来实现高效查找的方法。比如我需要查询一个数460,那么根据先前存储时所采取的映射关系就可以准确地得到

    相关 详解查找

    哈希表查找 定义 基本概念 实现方法 1、定义 > 哈希表查找又叫散列表查找,通过查找关键字不需要比较就可以获得需要记录的存储位置,它是通过在记

    相关 表的查找

    引入哈希表 前面查找方法共同特点:通过将关键字值与给定值比较,来确定位置。效率取决比较次数。 理想的方法是:不需要比较,根据给定值能直接定位记录的存储位置。 这样,