• 精准高效查找必备良方之-哈希表
  • 发布于 1周前
  • 55 热度
    1 评论
  • 李白羽
  • 21 粉丝 44 篇博客
  •   
哈希表我们几乎每天都在用,哈希表精准高效的查找,有点“万军丛中取敌将首级如探囊取物”的感觉。

但是它具体有哪些特性,如何去实现这些特性呢,又有多少同学会比较深入去理解呢?这篇博客我觉得写得不错,文章不长,循序渐进,并且在文末手撸了两种解决哈希碰撞的方法。无论新司机还是老司机,都值得读一读,温故而知新。

在对二叉树进行学习之后,相信每个人都会对于其精妙和最坏情况下仍然能保证线性对数级别时间复杂度的增删改查赞叹不已。其用来实现符号表(存储键值对)是一种较为高效的实现。

在没有学习哈希表之前,如果我告诉你,有一种数据结构,可以对键值对做到常数时间级别的增删改查,你一定不会相信。但是哈希表做到了,有这么高效的数据结构怎么能够不对其有了解呢?这次我们就来谈谈哈希表这种数据结构的魅力所在。

要了解哈希表,首先我们得来看一看这个哈希函数是什么东西。函数,其实就是一个映射,将所有在定义域的值映射到值域范围内。有时候我们需要对任意长度的信息,提取出信息的“指纹”,这个指纹在一定程度上反映了信息的内容,但是占用空间大大缩小。此时我们就要设计一个函数,其定义域为信息可能出现的所有空间,而值域则为需要”指纹”的大小。当这个函数可以用来完成上述用途时,这个函数就被称为哈希函数,哈希函数输出的结果被称为哈系值。一些著名的哈希函数有MD5、SHA-1、SHA-2等。

有了哈希函数,我们就可以利用哈希函数的映射,将任意长度的键映射到一个合理的区间内。试想创建一个数组,数组的下标为键的哈希值。数组的内容即为键值对的值,当我们已知键,要对值进行查找时,只需要对键取哈希,就可以在常数时间内获取到值的内容。

在哈希表的实现中,我们设计的哈希函数应该考虑的重点有如下几个

尽可能让输入中的每一位都对输出结果造成影响,减少碰撞的发生
计算尽量简单,否则哈希函数的计算也会对运行时间造成极大的影响
输出范围应当尽可能的可控
因为2的原因,我们不得不抛弃MD5和SHA这些著名的哈希函数,并且这些函数也很难自己控制输出的数据范围。哈希函数的好坏,决定了哈希表的效率。当然,实现哈希函数的方法不止一种,这里介绍比较常用的一种方法——除留余数法。

这里其实可以采用一种简单的运算来实现这个哈希函数——模运算。相信学过一点编程语言的朋友都知道,对某个数取模就是除以某个数求余数。假设我们需要0-97的哈希范围,无论多大的数,只需要对97取模,就可以获得一个在0-97之间的数,但是要注意,当我们采用的基数不是素数时,就会导致输出仅仅依赖于输入的后几位数,不满足1的情况发生。所以在使用这种方法作为哈希函数的过程中,应当选择合适的基数。

当我们有了哈系函数以后,就可以利用哈希函数类构造哈希表了。首先创建一个很大的数组array,数组的大小即为哈希函数的值域范围,然后在插入键值对时,直接使用array[hash(key)] = value的方式给数组赋值,当需要已知键,想要取出值的时候,同样采用array[hash(key)]即可。

下面我对哈希表的实现,供参考:

哈希碰撞的解决方法

读到这里,你可能已经发现了,哈希函数由于对信息进行了缩短,所以导致了有可能出现碰撞的存在,这就好像出现了两个名字一样的“张三”,当我们说,张三坐在某某位置上时,并不知道到底是哪个张三真正坐在位置上。这里有两种解决方案:拉链法和线性探测法

拉链法是将具有相同哈希值的函数放到一个链表里(整个哈希表呈现为一个链表的数组),用一个链表的数组来存放所有键值对。这种方法在最坏情况(哈希函数的返回值始终为常数)下会退化成链表。

而线性探测法则是采用两个数组,一个放键,一个放值,当即将保存的位置已经保存有数据的时候,从该位置开始往下遍历,直到遇到一个空的位置进行插入。这种方法在最坏情况下会退化成为数组。

两种方案对比,很难说哪种就一定跟高效,具体的实现中应当使用哪一种应该很据不同需求选取。

下面分别是线性探测和拉链法的哈希表的 C++实现:
  1. template <typename Key, typename Value> class LinearProbingHashST

  2. {

  3. private:

  4.  size_t size_;

  5.  size_t pair_counts_ = 0;

  6.  Key *key_arr_ ;

  7.  Value *value_arr_;

  8.  size_t gethash_(Key k);

  9.  Value get_ (Key k, size_t index);

  10.  Value put_ (Key k, Value val, size_t index);

  11.  void LinearProbingHashST_(size_t size);

  12. public:

  13.  LinearProbingHashST (size_t size);

  14.  LinearProbingHashST ();

  15.  virtual ~LinearProbingHashST ();

  16.  void put(Key k, Value val);

  17.  Value get(Key k);

  18.  void erase(Key k);

  19.  bool empty();

  20. };

  21. template <typename Key, typename Value>

  22. size_t LinearProbingHashST<Key, Value>::

  23. gethash_(Key k)

  24. {

  25.  return ( int(k) & 0x7FFFFFFF )% size_;

  26.  //return 1;

  27. }

  28. template <typename Key, typename Value>

  29. void LinearProbingHashST<Key, Value>::

  30. LinearProbingHashST_(size_t size)

  31. {

  32.  size_ = size;

  33.  key_arr_ = new Key[size_];

  34.  value_arr_ = new Value[size_]();

  35. }

  36. template <typename Key, typename Value>

  37. LinearProbingHashST<Key, Value>::

  38. LinearProbingHashST(size_t size)

  39. {

  40.  LinearProbingHashST_(size);

  41. }

  42. template <typename Key, typename Value>

  43. LinearProbingHashST<Key, Value>::

  44. LinearProbingHashST()

  45. {

  46.  LinearProbingHashST_(997);

  47. }

  48. template <typename Key, typename Value>

  49. LinearProbingHashST<Key, Value>::

  50. ~LinearProbingHashST()

  51. {

  52.  delete[] key_arr_;

  53.  delete[] value_arr_;

  54. }

  55. template <typename Key, typename Value>

  56. void LinearProbingHashST<Key, Value>::

  57. put(Key k, Value val)

  58. {

  59.  if (pair_counts_+1 >= size_)

  60.  {

  61.    return;

  62.  }

  63.  size_t i;

  64.  for (i = gethash_(k);key_arr_[i] != k;i = (i+1) % size_)

  65.  {

  66.    if (key_arr_[i] == (Key)NULL)

  67.    {

  68.      key_arr_[i] = k;

  69.      pair_counts_++;

  70.      break;

  71.    }

  72.  }

  73.  value_arr_[i] = val;

  74. }

  75. template <typename Key, typename Value>

  76. Value LinearProbingHashST<Key, Value>::

  77. get(Key k)

  78. {

  79.  for (size_t i = gethash_(k);key_arr_[i] != (Key) NULL;i = (i+1) % size_)

  80.  {

  81.    if (key_arr_[i] == k)

  82.    {

  83.      return value_arr_[i];

  84.    }

  85.  }

  86.  return (Value) NULL;

  87. }

  88. template <typename Key, typename Value>

  89. void LinearProbingHashST<Key, Value>::

  90. erase(Key k)

  91. {

  92.  for (size_t i = gethash_(k);key_arr_[i] != (Key) NULL;i = (i + 1) % size_)

  93.  {

  94.    if (key_arr_[i] == k)

  95.    {

  96.      key_arr_[i] = (Key) NULL;

  97.      value_arr_[i] = (Value) NULL;

  98.      pair_counts_--;

  99.      for (size_t j = i+1;key_arr_[j] != (Key) NULL;j = (j+1) % size_)

  100.      {

  101.        Key temp_k = key_arr_[j];

  102.        Value temp_val = value_arr_[j];

  103.        key_arr_[j] = (Key) NULL;

  104.        value_arr_[j] = (Key)NULL;

  105.        pair_counts_--;

  106.        put(temp_k, temp_val);

  107.      }

  108.      break;

  109.    }

  110.  }

  111. }

  112. template <typename Key, typename Value>

  113. bool LinearProbingHashST<Key, Value>::

  114. empty()

  115. {

  116.  return pair_counts_ == 0;

  117. }


用户评论