散列表,它是基于高速存取的角度设计的,也是一种典型的“空间换时间”的作法。顾名思义,该数据结构可以理解为一个线性表,但是当中的元素不是紧密排列的,而是可能存在空隙。php
散列表(Hash table,也叫哈希表),是依据关键码值(Key value)而直接进行訪问的数据结构。也就是说,它经过把关键码值映射到表中一个位置来訪问记录,以加快查找的速度。这个映射函数叫作散列函数,存放记录的数组叫作散列表。html
比方咱们存储70个元素,但咱们可能为这70个元素申请了100个元素的空间。70/100=0.7,这个数字称为负载因子。咱们之因此这样作,也是为了“高速存取”的目的。咱们基于一种结果尽量随机平均分布的固定函数H为每个元素安排存储位置,这样就可以避免遍历性质的线性搜索,以达到高速存取。但是由于此随机性,也一定致使一个问题就是冲突。所谓冲突,即两个元素经过散列函数H获得的地址一样,那么这两个元素称为“同义词”。这相似于70我的去一个有100个椅子的饭店吃饭。散列函数的计算结果是一个存储单位地址,每个存储单位称为“桶”。设一个散列表有m个桶,则散列函数的值域应为[0,m-1]。
解决冲突是一个复杂问题。冲突主要取决于:
(1)散列函数,一个好的散列函数的值应尽量平均分布。
(2)处理冲突方法。
(3)负载因子的大小。太大不必定就好,而且浪费空间严重,负载因子和散列函数是联动的。
解决冲突的办法:
(1)线性探查法:冲突后,线性向前试探,找到近期的一个空位置。缺点是会出现堆积现象。存取时,可能不是同义词的词也位于探查序列,影响效率。
(2)双散列函数法:在位置d冲突后,再次使用还有一个散列函数产生一个与散列表桶容量m互质的数c,依次试探(d+n*c)%m,使探查序列跳跃式分布。
常用的构造散列函数的方法算法
散列函数能使对一个数据序列的訪问过程更加迅速有效,经过散列函数,数据元素将被更快地定位:数组
1. 直接寻址法:取keyword或keyword的某个线性函数值为散列地址。即H(key)=key或H(key) = a•key + b,当中a和b为常数(这样的散列函数叫作自身函数)安全
2. 数字分析法:分析一组数据,比方一组员工的出生年月日,这时咱们发现出生年月日的前几位数字大致一样,这种话,出现冲突的概率就会很是大,但是咱们发现年月日的后几位表示月份和详细日期的数字区别很是大,假设用后面的数字来构成散列地址,则冲突的概率会明显减小。所以数字分析法就是找出数字的规律,尽量利用这些数据来构造冲突概率较低的散列地址。数据结构
3. 平方取中法:取keyword平方后的中间几位做为散列地址。ide
4. 折叠法:将keyword切割成位数一样的几部分,最后一部分位数可以不一样,而后取这几部分的叠加和(去除进位)做为散列地址。函数
5. 随机数法:选择一随机函数,取keyword的随机值做为散列地址,通常用于keyword长度不一样的场合。性能
6. 除留余数法:取keyword被某个不大于散列表表长m的数p除后所得的余数为散列地址。即 H(key) = key MOD p, p<=m。不只可以对keyword直接取模,也可在折叠、平方取中等运算以后取模。对p的选择很是重要,通常取素数或m,若p选的很差,easy产生同义词。
查找的性能分析this
散列表的查找过程基本上和造表过程一样。一些关键码可经过散列函数转换的地址直接找到,还有一些关键码在散列函数获得的地址上产生了冲突,需要按处理冲突的方法进行查找。在介绍的三种处理冲突的方法中,产生冲突后的查找仍然是给定值与关键码进行比較的过程。因此,对散列表查找效率的量度,依旧用平均查找长度来衡量。
查找过程当中,关键码的比較次数,取决于产生冲突的多少,产生的冲突少,查找效率就高,产生的冲突多,查找效率就低。所以,影响产生冲突多少的因素,也就是影响查找效率的因素。影响产生冲突多少有下面三个因素:
1. 散列函数是否均匀;
2. 处理冲突的方法;
3. 散列表的装填因子。
散列表的装填因子定义为:α= 填入表中的元素个数 / 散列表的长度
α是散列表装满程度的标志因子。由于表长是定值,α与“填入表中的元素个数”成正比,因此,α越大,填入表中的元素较多,产生冲突的可能性就越大;α越小,填入表中的元素较少,产生冲突的可能性就越小。
实际上,散列表的平均查找长度是装填因子α的函数,仅仅是不一样处理冲突的方法有不一样的函数。
了解了hash基本定义,就不能不提到一些著名的hash算法,MD5 和 SHA-1 可以说是眼下应用最普遍的Hash算法,而它们都是以 MD4 为基础设计的。那么他们都是什么意思呢?
这里简单说一下:
(1) MD4
MD4(RFC 1320)是 MIT 的 Ronald L. Rivest 在 1990 年设计的,MD 是 Message Digest 的缩写。它适用在32位字长的处理器上用快速软件实现--它是基于 32 位操做数的位操做来实现的。
(2) MD5
MD5(RFC 1321)是 Rivest 于1991年对MD4的改进版本号。它对输入仍以512位分组,其输出是4个32位字的级联,与 MD4 一样。MD5比MD4来得复杂,并且速度较之要慢一点,但更安全,在抗分析和抗差分方面表现更好
(3) SHA-1 及其它
SHA1是由NIST NSA设计为同DSA一块儿使用的,它对长度小于264的输入,产生长度为160bit的散列值,所以抗穷举(brute-force)性更好。SHA-1 设计时基于和MD4一样原理,并且模仿了该算法。
哈希表不可避免冲突(collision)现象:对不一样的keyword可能获得同一哈希地址 即key1≠key2,而hash(key1)=hash(key2)。所以,在建造哈希表时不只要设定一个好的哈希函数,而且要设定一种处理冲突的方法。可例如如下描写叙述哈希表:依据设定的哈希函数H(key)和所选中的处理冲突的方法,将一组keyword映象到一个有限的、地址连续的地址集(区间)上并以keyword在地址集中的“象”做为对应记录在表中的存储位置,这样的表被称为哈希表。
对于动态查找表而言,1) 表长不肯定;2)在设计查找表时,仅仅知道keyword所属范围,而不知道确切的keyword。所以,普通状况需创建一个函数关系,以f(key)做为keyword为key的录在表中的位置,一般称这个函数f(key)为哈希函数。(注意:这个函数并不必定是数学函数)
哈希函数是一个映象,即:将keyword的集合映射到某个地址集合上,它的设置很是灵活,仅仅要这个地址集合的大小不超出赞成范围就能够。
现实中哈希函数是需要构造的,并且构造的好才干使用的好。
那么这些Hash算法究竟有什么用呢?
Hash算法在信息安全方面的应用主要体现在下面的3个方面:
(1) 文件校验
咱们比較熟悉的校验算法有奇偶校验和CRC校验,这2种校验并无抗数据篡改的能力,它们必定程度上能检測并纠正传输数据中的信道误码,但却不能防止对数据的恶意破坏。
MD5 Hash算法的"数字指纹"特性,使它成为眼下应用最普遍的一种文件完整性校验和(Checksum)算法,很多Unix系统有提供计算md5 checksum的命令。
(2) 数字签名
Hash 算法也是现代password体系中的一个重要组成部分。由于非对称算法的运算速度较慢,因此在数字签名协议中,单向散列函数扮演了一个重要的角色。 对 Hash 值,又称"数字摘要"进行数字签名,在统计上可以以为与对文件自己进行数字签名是等效的。而且这种协议还有其它的长处。
(3) 鉴权协议
例如如下的鉴权协议又被称做挑战--认证模式:在传输信道是可被侦听,但不可被篡改的状况下,这是一种简单而安全的方法。
文件hash值
MD5-Hash-文件的数字文摘经过Hash函数计算获得。不管文件长度怎样,它的Hash函数计算结果是一个固定长度的数字。与加密算法不一样,这一个Hash算法是一个不可逆的单向函数。採用安全性高的Hash算法,如MD五、SHA时,两个不一样的文件差点儿不可能获得一样的Hash结果。所以,一旦文件被改动,就可检測出来。
Hash函数还有另外的含义。实际中的Hash函数是指把一个大范围映射到一个小范围。把大范围映射到一个小范围的目的每每是为了节省空间,使得数据easy保存。除此之外,Hash函数每每应用于查找上。因此,在考虑使用Hash函数以前,需要明确它的几个限制:
1. Hash的主要原理就是把大范围映射到小范围;因此,你输入的实际值的个数必须和小范围至关或者比它更小。否则冲突就会很是多。
2. 由于Hash逼近单向函数;因此,你可以用它来对数据进行加密。
3. 不一样的应用对Hash函数有着不一样的要求;比方,用于加密的Hash函数主要考虑它和单项函数的差距,而用于查找的Hash函数主要考虑它映射到小范围的冲突率。
应用于加密的Hash函数已经探讨过太多了,在做者的博客里面有更具体的介绍。因此,本文仅仅探讨用于查找的Hash函数。
Hash函数应用的主要对象是数组(比方,字符串),而其目标一般是一个int类型。下面咱们都依照这样的方式来讲明。
通常的说,Hash函数可以简单的划分为例如如下几类:
1. 加法Hash;
2. 位运算Hash;
3. 乘法Hash;
4. 除法Hash;
5. 查表Hash;
6. 混合Hash;
如下具体的介绍以上各类方式在实际中的运用。
一 加法Hash
所谓的加法Hash就是把输入元素一个一个的加起来构成最后的结果。标准的加法Hash的构造例如如下:
这里的prime是随意的质数,看得出,结果的值域为[0,prime-1]。
二 位运算Hash
这类型Hash函数经过利用各类位运算(常见的是移位和异或)来充分的混合输入元素。比方,标准的旋转Hash的构造例如如下:
先移位,而后再进行各类位运算是这样的类型Hash函数的主要特色。比方,以上的那段计算hash的代码还可以有例如如下几种变形:
三 乘法Hash
这样的类型的Hash函数利用了乘法的不相关性(乘法的这样的性质,最有名的莫过于平方取头尾的随机数生成算法,尽管这样的算法效果并很差)。比方,
jdk5.0里面的String类的hashCode()方法也使用乘法Hash。只是,它使用的乘数是31。推荐的乘数还有:131, 1313, 13131, 131313等等。
使用这样的方式的著名Hash函数还有:
以及改进的FNV算法:
除了乘以一个固定的数,常见的还有乘以一个不断改变的数,比方:
尽管Adler32算法的应用没有CRC32普遍,只是,它多是乘法Hash里面最有名的一个了。关于它的介绍,你们可以去看RFC 1950规范。
四 除法Hash
除法和乘法同样,相同具备表面上看起来的不相关性。只是,因为除法太慢,这样的方式差点儿找不到真正的应用。需要注意的是,咱们在前面看到的hash的 结果除以一个prime的目的仅仅是为了保证结果的范围。假设你不需要它限制一个范围的话,可以使用例如如下的代码替代”hash%prime”: hash = hash ^ (hash>>10) ^ (hash>>20)。
五 查表Hash
查表Hash最有名的样例莫过于CRC系列算法。尽管CRC系列算法自己并不是查表,但是,查表是它的一种最快的实现方式。如下是CRC32的实现:
查表Hash中有名的样例有:Universal Hashing和Zobrist Hashing。他们的表格都是随机生成的。
六 混合Hash
混合Hash算法利用了以上各类方式。各类常见的Hash算法,比方MD五、Tiger都属于这个范围。它们通常很是少在面向查找的Hash函数里面使用。
七 对Hash算法的评价
http://www.burtleburtle.net/bob/hash/doobs.html 这个页面提供了对几种流行Hash算法的评价。咱们对Hash函数的建议例如如下:
1. 字符串的Hash。最简单可以使用主要的乘法Hash,当乘数为33时,对于英文单词有很是好的散列效果(小于6个的小写形式可以保证没有冲突)。复杂一点可以使用FNV算法(及其改进形式),它对于比較长的字符串,在速度和效果上都不错。
public override unsafe int GetHashCode()
{//微软System.String 字符串哈希算法
fixed (char* str= ((char*) this))
{
char* chPtr = str;
intnum = 0x15051505;
intnum2 = num;
int* numPtr = (int*) chPtr;
for (inti = this.Length; i > 0; i -= 4)
{
num = (((num << 5) + num) + (num >> 0x1b)) ^ numPtr[0];
if (i <= 2)
{
break;
}
num2 = (((num2 << 5) + num2) + (num2 >> 0x1b)) ^ numPtr[1];
numPtr += 2;
}
return (num + (num2 * 0x5d588b65)); }}