深刻理解数据结构之散列表、散列、散列函数

           前言

                           笔者之前对散列是什么?哈希又是什么?何谓散列表?散列函数又是个什么东东比较的迷惑。java

                    经过看一些书,查找一些资料总算是有一些眉目了,现将相关的知识与体会记录下来。留待往后算法

                    的再学习!数组

           基本概念

                          散列表(Hash table,也叫哈希表),是根据关键字(key value)而直接进行访问的数据结构。安全

                    说的具体点就是它经过吧key值映射到表中的一个位置来访问记录,从而加快查找的速度。数据结构

                          实现key值映射的函数就叫作散列函数app

                          存放记录的数组就就叫作散列表dom

                          实现散列表的过程一般就称为散列(hashing),也就是常说的hashide

                散列

                           这里的散列的概念不只限于数据结构了,在计算机科学领域中,散列-哈希是一种对信息的函数

                     处理方法,经过某种特定的函数/算法(散列函数/hash()方法)将要检索的项与用来检索的索引-源码分析

                     -( 散列值)关联起来,生成一种便于搜索的数据结构--散列表。

                           现在,因为散列算法所计算的散列值 具备不可逆(没法逆向演算会原来的数值)的性质,

                     所以散列算法普遍的运用于加密技术。

                           散列的运用:
                                   一、加密散列
                                        在信息安全领域使用
                                   二、散列表
                                         一种使用散列函数将键名和键值关联起来的数据结构
                                   三、关联数组
                                         一种经常使用散列表来实现的数据结构
                                   四、几何散列
                                         寻找相同或类似的几何形状的一种有效方法 

                  散列函数

                              经过上面能够知道,散列技术的实现是基于散列函数的。这里对散列函数进行一个较深

                         入的理解。  前面就知道了散列函数--哈希函数就是完成key值与位置的映射。通常说来key

                         以字符 串的形式居多,位置也就是一个数值。

                              能够看出散列函数就像是实现信息的压缩,把消息字符 串压缩成数值摘要,是数据量

                         变小,格式得以固定下来。

                              散列函数的工做原理图:

                    

                               不过须要注意的是key值和通过散列函数处理以后的散列值并非惟一对应的,

                        这就形成了不一样的key值具备相同的索引位置,这种现象叫作散列碰撞、也称其为哈希冲突。

                        对于hash冲突的解决办法,将在后面予以总结。

                               至于散列函数的具体实现,有不少加密技术都有十分nice的实现,这里咱们看看java中

                        HashMap的hash()方法实现就能够了。HashMap采用的是内部哈希技术实现的,其中

                        hash()方法就是散列函数,完成key值到数组索引位置的映射。                    

 /**      * Retrieve object hash code and applies a supplemental hash function to the      * result hash, which defends against poor quality hash functions.  This is      * critical because HashMap uses power-of-two length hash tables, that      * otherwise encounter collisions for hashCodes that do not differ      * in lower bits. Note: Null keys always map to hash 0, thus index 0.      */     final int hash(Object k) {         int h = 0;         if (useAltHashing) {             if (k instanceof String) {                 return sun.misc.Hashing.stringHash32((String) k);             }             h = hashSeed;         }          h ^= k.hashCode();          // This function ensures that hashCodes that differ only by         // constant multiples at each bit position have a bounded         // number of collisions (approximately 8 at default load factor).         h ^= (h >>> 20) ^ (h >>> 12);         return h ^ (h >>> 7) ^ (h >>> 4);     } 
                            上述代码就是HashMap中散列函数的具体实现。JDK1.7

                        这里笔者对经常使用的散列算法作一个展现:

               

               散列表

                                 在理解了上述散列\散列函数的概念以后咱们正式的进入到散列表的学习.

                             一个通俗的例子是,为了查找电话簿中某人的号码,能够建立一个按照人名首字母顺序排列

                            的表(即创建人名 x 到首字母 F(x) 的一个函数关系),在首字母为 W 的表中查找“王”姓的电

                            话号码,显然比直接查找就要快得多。这里使用人名做为关键字,“取首字母”是这个例子中散

                            列函数的函数法则 F(),存放首字母的表对应散列表。关键字和函数法则理论上能够任意肯定。

                    散列函数的构造

                                对于散列表这种数据结构来讲,其散列函数的构造是十分关键的,散列函数实现了key的

                           映射,而且访问记录能够更快的被定位。

                                通常来讲散列函数的构造基于两个标准:简单、均匀

                                简单指散列算法简单快捷,散列值生成简单。

                                均匀指对于key值集合中的任一关键字,散列函数可以以均与的几率映射到数组的任一一个

                           索引位置上,这样可以减小散列碰撞。

                                散列函数构造方法:

                             一、直接地址法:

                                              直接取key值或者key值的某个线性函数值做为散列地址。即hash(k)=k

                                       或者hash(k)=a*k+b。

                                             Tips: 简单的思考一下这种方式就能够知道,这种方式基本不会存在哈希冲突,

                                      不过事 先咱们应该知道key集合的大小,并且使用线性函数值做为散列地址的话,

                                      很大程度上形成了 空间的浪费。hash(k)=k这种方式更加的鸡肋不必,以这种方式

                                      散列还不如直接数组索引。

                              二、数字分析法:

                                             所谓的数字分析法就是假设关键字key是以r为基的数,而且hash表中可能出现的

                                    关键字都是事先知道的,则可取关键字的若干数位组成hash地址。

                                            Tips:这种方式极度不灵活,限制太多。

                              三、平方取中法:

                                            先经过求关键字的平方值扩大相近数的差异,而后根据表长度取中间的几位数做为

                                    散列函数值。

                                            Tips:这种方式中间的几位数都和关键字的没一位都有关,产生的散列地址较为的

                                    均匀。

                               四、折叠法:

                                           将关键字分割成相同的几位数(最后一位可不一样),而后去这几部分的叠加和。折叠法

                                   通常是和除留余法一块儿使用的。

                                五、除留余法:

                                          取关键字被某个不大于散列表表长 m 的数 p 除后所得的余数为散列地址。即 hash(k)

                                    = k mod p, p < m。不只能够对关键字直接取模,也可在折叠法、平方取中法等运算以后

                                    取模。对 p 的选择很重要,通常取素数或 m ,若 p 选择很差,容易产生碰撞。

                                 六、随机法:

                                           h(key)=random(key)   

                                        其中random为伪随机函数,但要保证函数值是在0到m-1之间。 

                               总结:在上述的方法中,三、四、5三种方法的结合使用方式较好,在JDK之前的版本就是使用

                                        的方法5。

                   哈希冲突

                               经过上面的学习中,咱们知道散列函数获得的key -  索引位置 并非惟一对应的,可能形成

                           不一样的key值对应相同的索引位置。这是咱们应该解决的问题。实际的解决方法通常以下:

                         一、分离链接法:

                               首先看看分离链接法,说白了这种方式就是链表数组的方式,将散列到同一个值得全部元素

                          保存在一个表中,产生相同的一个值在散列表中使用链表的形式存储。哈希冲突的位置就是链表

                          的开始位置。在JKD中HashMap就是这种方式解决哈希冲突的!

                         

                            HashMap中冲突处理代码以下                 

 for (Entry<K,V> e = table[i]; e != null; e = e.next) {             Object k;             if (e.hash == hash && ((k = e.key) == key || key.equals(k))) {                 V oldValue = e.value;                 e.value = value;                 e.recordAccess(this);                 return oldValue;             }         }
                                 详细的状况能够看看笔者之前基于HashMap源码分析的文章:

                                    http://blog.csdn.net/kiritor/article/details/8885961

                             这里咱们定义一个填装因子L,它表示散列表中实际的元素个数和数组的大小之比,

                       由此咱们能够很容易的获得查找一个元素的时间为计算散列值的常数时间+ 遍历链表

                       的时间。一次查找时间大约为1+2/L。由此说明,散列表的大小并非影响查找的关键,

                       关键在于L,保证L近似为1是十分有效的。

                       二、开放地址法

                             分离链接法的缺点在于使用了链表,因为给新的单元分配地址耗费时间,形成算法速度

                         较慢,解决的方法就是开放地址法,在开放地址法中较为经常使用的有两种:

                              线性探测法、平方探测法。

                             开放地址法:        

                              hash_i=(hash(key) + d(i)) mod m, i=1,2...k\,(k < m-1),其中hash(key)为散列函数,

                             m为散列表长,d(i)为增量序列,i为已发生碰撞的次数。增量序列可有下列取法:

                             d(i)=1,2,3...(m-1) 称为 线性探测;即 d(i)=i ,或者为其余线性函数。至关于逐个探测

                                                  存放地址的表,直到查找到一个空单元,把散列地址存放在该空单元。
                             d(i)=1^2,  2^2,3^2... k^2 (k < m/2) 称为 平方探测。相对线性探测,至关于发生碰

                             撞时探测间隔 d(i)=i^2 个单元的位置是否为空,若是为空,将地址存放进去。
                             d(i)=伪随机数序列,称为 伪随机探测。 

                             线性探测法

                                下面笔者将以一个实例演示线性探测的过程,进而分析线性探测的特色,引出平方探测

                                关键字为{89,18,49,58,69}插入到一个散列表中的状况。此时线性探测的方法是取d(i)=i。

                                并假定取关键字除以 10 的余数为散列函数法则。

                                    

                                                 一、开始时hash(89)=9无冲突,直接插入;

                                                 二、hash(18)=8无冲突,直接插入;

                                                 三、hash(49)=9冲突了,开放地址,将49放入下一个空闲地址0

                                                 四、hash(58) =8冲突了,开放地址,将58放入9冲突 ,放入0冲突、放入1

                                                 五、hash(69) =9冲突,开放地址,将69放入0冲突,放入1冲突,放入2

                               Tips:思考其缺点!

                                    线性探测的方式十分简单,明白,每次插入老是可以找到一个地址,可是慢慢会

                                造成一个区块,其结果称为一次汇集。任何关键字需探测愈来愈多的次数才能解决

                               冲突,且完成以后由简介的增大了区块。当填装因子>0.5时,这种方式就不是个好

                               的方法了!

                             平方探测法:

                                      使用平方探测法能够解决线性探测的一次汇集问题。通常选择d(i)=i^2.。

                                  至于其具体的步骤读者能够按照上面的实例自行的模拟一下。

                                     这种方式会出现二次汇集的状况:散列到同一位置的哪些元素将探测相同的备选

                                  单元。

                           三、双散列、再散列

                                 对于双散列和再散列的方式笔者这里就不在多提了。读者能够查阅下相关的资料。

                                总结:对于散列表的实现新手没必要太过在乎,关键在于理解散列相关的概念。了解并

                         掌握散列函数的做用及通常的实现方式。了解通常hash冲突和经常使用解决办法。

                                 到这儿就结束了,但愿你们看的开心,玩的愉快,端午节快乐。

                                                                                                                                             By    Kiritor

                                                                                                                                             写于2013 端午

相关文章
相关标签/搜索