第7篇:C++ 哈希表--散列函数

哈希表是一个非常强大的数据结构,我本篇系列的文章,我们会讲述以下内容

  • 什么是哈希表(Hash table),什么是散列函数(hash function)?
  • 散列函数的属性
  • 解决散列算法的冲突问题,其中流行的方案是开放寻址和单独链接,因此我们将要衍生讨论线性探测和二次探测的完成方式,本篇后续会以示例讲解他们的工作原理
  • Separate chaining 实现的细节

什么是哈希表

哈希表就是一种使用散列算法(Hashing)提供从键到值的映射的数据结构,下面就是键值对的映射示例,我们只要通过一个唯一可识别的名称即可映射到对应该名字的年龄,像这样的映射关系,我们叫做键值对(key-value pairs)

ss8.png

因此在哈希表中,键可以是任意的数据类型,只要它是键中是唯一的即可映射到一个值,而值是不必唯一的。

哈系表的用途

哈系表经常用于分类统计和跟踪键中的对象出现的频率

理解哈希表中的键值对(key-value pairs)构成,首选需要了解什么散列函数

散列函数

简单地说,散列函数h(x)是将键x映射到固定范围内的整数的函数,我们看看看一些简单的哈希函数的示例


ss8.png

这是一个不好的示例,因为键部分插入该示例函数会得到产生多个相同的值,例如h(4)和h(9),h(8)和h(12)都分别映射到同一个数字,这是没问题的,我们不仅可以在证书上定义任意的散列函数,还可以在任意对象上定义任意哈希函数,这使得这一函数非常强大

例如对于字符窜,我们可以定义一个和ascii码相关的散列函数h(x)的伪代码

function H(str):
    sum:=0
    for c in s:
        sum+=ascii(c)
    return sum mod 50

这个散列函数所做的就是对字符串中的字符ASCII值求和,然后在结尾处做取模运算,所以对于任何给定的字符串,它只输出一个数字,这就是一个散列函数,如下图不同的字符串传入H(x)会得到随机散列值


ss8.png

有了前面的基础,我们讨论一个更添加实际开发中的例子,例如我们有一个person的关系表,通常person表中每条记录有三个字段Name,Age,Sex构成一个person对象的基本特征信息,散列函数是的算法可以是任意可能的算法组合,但最终要的是散列函数在计算传入对象的散列值的的时间开销能能维持O(1)是最为理想的


ss8.png

我们希望为每个person对象的特征定义散列函数H(p),能够映射到集合{0,1,2,3,4,5,6},这次为了简化示例,我们使用python代码定义了下面关于Person对象的散列函数hash(p),该散列函数的起始计算元素是Person.age,接着Person.name,接着是Person.sex,最后是以除数7做取模运算

class Person:
    def __init__(self,name,age,sex):
        self.name=name
        self.age=age
        self.sex=sex
    #end-def
#end-class

def hash(p):
    hs=p.age
    hs=hs+len(p.name)
    if p.sex=='M':
        hs+=1
    return hs % 7
#end-def

p1=Person('Golden',24,'M')
p2=Person('Cally' ,32,'F')
p3=Person('Mady'  ,27,'F')
p4=Person('Cat'   ,49,'M')
p5=Person('Zucy'  ,15,'M')

pList=[p1,p2,p3,p4,p5]

for p in pList:
    print("Person对象:{} - hash:{}".format(p.name,hash(p)))

上面代码的每个Person对象的散列值的运算结果如下表


ss8.png

散列函数的属性

  1. 如果h(x)≠h(y),那么x≠y 也就是说,x和y是两个不同的对象
  2. 如果h(x)=h(y),那么x=y
    也就是说,如果我们有一个散列函数,并且两个对象分别由x和y并且它们的散列值相等,那么这两个对象可能相等,我们不确定是否必须显式地检测x和y,三列函数告诉我们在x和y散列值相同的位置发生了冲突

由上面的散列函数的属性引申出一个问题,
如何利用这个优势来加快对象比较?答案就是我们在初始化对象x和y之初应计算出两个对象的散列值,比较x,y的散列值的时间消耗要优于显式遍历x,y对象并比较他们的属性

在下面这个例子中,考虑尝试确定两个大文件是否具有相同的内容的问题

如果我们预编译了file1和fil2,首先应该比较这些哈希值,因为比较哈希值是O(1),如果可能,我们不想直接打开任何一个文件。逐个字节地比较它们的内容会很慢,尽管我们 如果H(file1)= H(file2),则可能必须这样做。

注意:用于文件的哈希函数比用于哈希表的哈希函数更复杂,对于文件,我们使用所谓的加密哈希函数,也称为校验和

  1. 散列函数的另外一个属性就是其散列值必须绝对确定的
    这意味这如果y=h(x),那么h(x)必须总能产生y并且不可能是另外的值(字符串或整数),这可能很明显,但是对于哈希函数的功能至关重要

那么什么是非确定性呢?这个很容易理解
读者可以看看上文的示例的散列函数,我们基本上在散列函数内部定义一个基本数据类型的局部变量,这是散列函数绝对确定性的关键。因为它们都没有用到全局变量作为计算散列值的数据源,也就是非确定性的函数引用了全局变量会造成每个传入对象的散列值的不确定性

冲突

散列冲突(Hash Collision)就是指当两个对象m,n传入某个散列函数h(x)返回了相同的散列值,即h(m)=h(n),在哈系表插入操作中我们简称冲突(collison),上面的示例中,都不同程度地出现了冲突



在实现哈希表时,我们尽可能使散列函数减少散列冲突的次数,这样做的原因是为了使哈希函数值域中的所有值都能够命中,以便可以均匀地填充到表中

什么是可散列化

现在,我们可以回答有关允许在哈希表中使用的键类型的核心问题,什么使T类型的键(Key)可散列化?
答: 因为我们要在哈希表的实现中使用散列函数,所以我们需要散列函数是绝对确定性的。为了强制这种行为,我们要求哈希表中使用的键(Key)是不可变的数据类型。如果T类型的键是不可变的,并且我们为T类型的所有键k定义了哈希函数H(k),那么我们称之为键类型T是可散列(hashable)

哈希表如何工作?

理想情况下,我们希望对放入哈希表中的数据有非常快的插入,查找和删除时间。散列函数为我们提供了在哈希表中查找对应元素的索引,值得注意的是,仅当您具有算法设计良好的哈希函数时,我们可以使用哈希函数在O(1)时间内将所有这些操作,以此作为索引到哈希表的方式,而哈希表只是数组的一个奇特的称呼罢了。
如下图是一个哈希表的逻辑图,每次向表中插入元素时,需通过散列函数计算被插入元素的散列值,对应的散列值就是表中的索引,查找元素也时,同理,只需计算键对应的散列值就能找到表中对应索引的元素。随着表中插入元素的增多,冲突的概率就大大增加,例如键Golden和键Mady的数据项是冲突的,


那么在哈希表中有许多方案解决散列冲突,流行的解决方案主要有

  • 开放寻址(open addressing):此法的思想是通过探测并在哈希表中查找与我们最初进行散列对应的位置在偏移后的另一个位置来处理哈希冲突。这里你如果不明白的话,我打个通俗的比喻,比如广东人去酒楼就早茶需要提前预订位置的,比如你定的桌子的编号和其他人冲突了,那么酒楼经理就会为你另外安排另外一个编号桌子一样。
  • 单独链表(separate chainning):此法的思想是通过维护数据结果来处理冲突,通常是一个链表,链表内包含所有为某个特定散列值的所有不同的键值对

哈希表的时间复杂度

只有在具有良好设计的均匀散列函数时,哈希表在平均情况下的下列操作的常量时间行为才能成立,
均匀散列函数:就是为任意的传入对象生成的散列值,任意两个传入对象发生冲突的可能性最低的散列函数,事实上是散列冲突是无法避免的,但如果你在一次固定次数的插入操作中产生冲突的次数超过你插入次数的10%的话(这是我个人所定的测试基准),你就要审视你写的散列函数就需要重构了

小结

我们本篇只要介绍了哈希表的一个最核心的主体,那就是散列函数(Hash Function)它是构成整个哈系表,且有别于普通数组的重要基石。下一篇我们会详细介绍避免散列冲突的技术--开放寻址(Open Addressing)的基本原理

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351

推荐阅读更多精彩内容