编程进阶网编程进阶网
  • 基础组成体系
  • 程序编程原理
  • 异常和IO系统
  • 六大设计原则
  • 设计模式导读
  • 创建型设计模式
  • 结构型设计模式
  • 行为型设计模式
  • 设计模式案例
  • 面向对象思想
  • 基础入门
  • 高级进阶
  • JVM虚拟机
  • 数据集合
  • Java面试题
  • C语言入门
  • C综合案例
  • C标准库
  • C语言专栏
  • C++入门
  • C++综合案例
  • C++专栏
  • HTML
  • CSS
  • JavaScript
  • 前端专栏
  • Swift
  • iOS入门
  • 基础入门
  • 开源库解读
  • 性能优化
  • Framework
  • 方案设计
  • 媒体音视频
  • 硬件开发
  • Groovy
  • 常用工具
  • 大厂面试题
  • 综合案例
  • 网络底层
  • Https
  • 网络请求
  • 故障排查
  • 专栏
  • 数组
  • 链表
  • 栈
  • 队列
  • 树
  • 递归
  • 哈希
  • 排序
  • 查找
  • 字符串
  • 其他
  • Bash脚本
  • Linux入门
  • 嵌入式开发
  • 代码规范
  • Markdown
  • 开发理论
  • 开发工具
  • Git管理
  • 百宝箱
  • 开源协议
  • 技术招聘
  • 测试经验
  • 职场提升
  • 技术模版
  • 关于我
  • 目标清单
  • 学习框架
  • 育儿经验
  • 我的专栏
  • 底层能力
  • 读书心得
  • 随笔笔记
  • 职场思考
  • 中华历史
  • 经济学故事
  • 基础组成体系
  • 程序编程原理
  • 异常和IO系统
  • 六大设计原则
  • 设计模式导读
  • 创建型设计模式
  • 结构型设计模式
  • 行为型设计模式
  • 设计模式案例
  • 面向对象思想
  • 基础入门
  • 高级进阶
  • JVM虚拟机
  • 数据集合
  • Java面试题
  • C语言入门
  • C综合案例
  • C标准库
  • C语言专栏
  • C++入门
  • C++综合案例
  • C++专栏
  • HTML
  • CSS
  • JavaScript
  • 前端专栏
  • Swift
  • iOS入门
  • 基础入门
  • 开源库解读
  • 性能优化
  • Framework
  • 方案设计
  • 媒体音视频
  • 硬件开发
  • Groovy
  • 常用工具
  • 大厂面试题
  • 综合案例
  • 网络底层
  • Https
  • 网络请求
  • 故障排查
  • 专栏
  • 数组
  • 链表
  • 栈
  • 队列
  • 树
  • 递归
  • 哈希
  • 排序
  • 查找
  • 字符串
  • 其他
  • Bash脚本
  • Linux入门
  • 嵌入式开发
  • 代码规范
  • Markdown
  • 开发理论
  • 开发工具
  • Git管理
  • 百宝箱
  • 开源协议
  • 技术招聘
  • 测试经验
  • 职场提升
  • 技术模版
  • 关于我
  • 目标清单
  • 学习框架
  • 育儿经验
  • 我的专栏
  • 底层能力
  • 读书心得
  • 随笔笔记
  • 职场思考
  • 中华历史
  • 经济学故事
  • 01.Hash算法待完善

目录介绍

  • 01.散列表性能思考
  • 02.设计散列函数
  • 03.装载因子大小探索
  • 04.避免低效地扩容
  • 05.冲突解决方法
  • 06.工业级散列表案例
  • 07.看思考题
  • 08.内容小结

01.散列表性能思考

  • 散列表的查询不能笼统的说是O(1)。它跟散列函数、装载因子、散列冲突等都有关系。如果散列表设计得不好,或者装载因子过高,都可能导致散列冲突发生的概率升高,查询效率下降。
  • 在极端的情况下,有些恶意攻击,还可能通过精心构造的数据,使得所有的数据经过散列之后,都散列到同一个槽里。如果我们使用的是基于链表的冲突解决方法,那这个时候,散列表就退化为链表,查询的时间复杂度就从O(1)急剧退化这O(n)。
  • 如果散列表中有 10 万个数据,退化后的散列表查询的效率就下降了10万倍。更直接点说,如果之前运行 100 次查询只需要 0.1 秒,那现在就需要 1万秒。这样就有可能因为查询操作消耗大量 CPU 或者线程资源,导致系统无法响应其他请求,从而达到拒绝服务攻击(DoS)的目的。这也就是散列表碰撞攻击的基本原理。
  • 如何设计一个可以应对各种异常情况的工业级散列表,来避免散列冲突的情况下,散列表功能急剧下降,并且能抵抗散列碰撞攻击?

02.设计散列函数

  • 散列函数设计的好坏,决定了散列冲突的概率大小,也直接散列表的性能。那什么才是好的散列函数呢?
  • 首先,散列函数的设计不能太复杂。
    • 过于复杂的散列函数,势必会消耗很多计算的时间,也就间接影响到散列表的性能。其次,散列函数生成的值要尽可能随机并且均匀分布,这样才能避免或者最小化散列冲突,而且即便出现冲突,散列到每个槽时的数据也会比较平均,不会出现某个槽内数据特别多的情况。
  • 实际工作中,我们来需要综合考虑各种因素。这些因素有关键字的长度、特点、分布、还有散列表的大小等。散列函数各式各样,举几个例子。
    • 第一个例子就是上一节学生运动会的例子。我们通过分析参赛编号的特征,把编号中的后两位作为散列值。这种散列函数的设计方法,我们一般叫作“数据分析法”。
    • 第二个例子就是如何实现Word拼写检查功能。这里面的散列函数,我们就可以这样设计:将单词中的每个字母的ASCII码值“进位”相加,然后再跟散列表的大小求余、取模,作为散列值。
  • 实际上,散列函数的设计方法还有很多,比如直接寻址法、平方取中法、折叠法、随机数法,这些了解就行了。

03.装载因子大小探索

  • 装载因子过大了怎么办?
    • 对于没有频繁插入和删除的静态数据集合来说,很低容易根据的特点、分布等,设计出完美的、极少总被的散列函数,因为毕竟之间都是已知的 。
  • 对于动散列表来说,数据集合是频繁变动的,我们事先无法预估将要加入的数据个数,所以我们也无法事先申请一个足够大的散列表。随着数据慢慢加入,装载因子就会慢慢变大。当大到一定程度之后,散列冲突就会变得不可接受。这个时候,该如何处理呢?
  • 我们或以进行动态扩容,重新申请一个更大的散列表,将数据搬移到这个新的散列表中。假设每次扩容我们都申请一个原来二八现象表大小两倍的空间。如果原来散列表的装载因子是0.8,那经过扩容之后,新的表的装载因子就下降到原来的一半,成了0.4。
  • 针对散列表的扩容,数据搬移操作要复杂的多,因为散列表的大小变了,数据的存储位置也变了,所以我们需要通过散列函数重新计算每个数据的存储位置。
  • 如下图,在原来的散列中,21这个放下过去原来存储在下标为0的位置,搬移到新的散列表中,存储在下标为7的位置。
    • image
      image
  • 支持动态扩容的散列表,插入操作的时间复杂度是多少呢?插入一个数据,最好情况个,不需要扩容,最好时间复杂度是O(1)。最坏情况下,散列表装载因子过高,启动扩容,我们需要重新申请内存空间,重新计算哈希位置,并且搬移,所以时间复杂度是O(n)。均摊情况下,时间复杂度接近最好情况,就是O(1)。
  • 前面讲到,当散列表的装载因子超过某个阈值时,就需要进行扩容。装载因子阈值需要选择得当。如果太大,会导致冲突过多;如果太小,会导致内存浪费严重。
  • 装载因子阈值的调协要权衡时间、空间复杂度。如果内存空间不紧张,对执行效率要求很高,可以降低装载因子的阈值;相反,如果内存空间照张,对执行效率要求又不高,可增加装载因子的值。

04.避免低效地扩容

  • 刚刚分析到,大部分情况下,动态扩容的散列表插入一个数据都很快,但在特殊情况下,当装载因子已经达到阈值,需要进行扩容,再插入数据。这个时候,插入数据就会变得很慢,甚至会无法接受。
  • 我举一个极端的例子,如果散列表当前的大小是1GB,要想扩容到原来的两位大小,那就需要对1GB的数据重新计算哈希值,并且原来的散列表搬移到新的散列表,听起来就很耗时吧!
  • 如果我们的业务代码直接服务于用户,尽管大部分情况下,插入一个数据的操作都很快,但是,极个别非常慢的插入操作,也会让用户崩溃。这个时候,“一次性”扩容的机制就不合适了。
  • 这样的情况,我们可以将扩容操作插在插入操作的过程上,分批完成。当装载因子触达阈值之后,我们只申请新空间,但并不将老的数据搬移到新的散列表中。
  • 当有新数据要插入时,我们将新数据插入到新的散列表中,并且从老的散列表中拿出一个数据放入到新的散列表。每次插入一个数据到散列表,我们都重复上面的过程。经过多次插入操作之后,老的散列表中的就一点一点全部搬移到新的散列表中了。
  • image
    image
  • 这期间的查询操作怎么来做呢?这了兼容,我们先从新散列中查找,如果没有,再去老散列表中查找。
  • 通过这样的均摊的方法,将一次性扩容的代价,均摊到多次插入操作中,就避免了一次性扩容耗时过多的情况。这种实现方式,任何情况下,插入一个数据的时间复杂度都是O(1)。

05.冲突解决方法

  • 上一节我们讲了两个主要的散列冲突的解决方法,开放寻址法和链表法。这两种冲突解决办法在实际的软件开发中都非常常用。比如Java中LinkHashMap就采用了链表法解决冲突,TreadLocalMap是通过线性探测的开放寻址解决冲突的。
  • 1、开放寻址法
    • 开放寻址法不像链表法,需要拉拉很多链表。散列表中的数据存储在数组中,可以有效地利用CPU缓存加快查询速度。而且,这种方法实现的散列表,序列化起来比较简单。链表法包吃住指针,序列化起来就没那么容易。你可不要小看序列化,很多场合都会用到的。
    • 开放寻址解决冲突的散列表,删除的时候比较麻烦,需要特殊标记已删除掉的数据。而且,在开放寻址法中,所有的数据都存储在一个数组中,比起链表来说,冲突的代价更高。所以,使用开放寻址法解决冲突的散列表,装载因子的上限不能太大。这也导致这种方法比链表法更浪费内存空间。
    • 所以,当数据量比较小、装载因子小的时候,适合采用开放寻址法。这也是Java中的ThreadLocalMap使用开放寻址法解决散列冲突的原因。
  • 2、链表法
    • 首先,链表法对内存的利用率比开放寻址法要高。因为链表结点可以在需要的时候再创建,并不需要像开放寻址法那样事先申请好。它对装载因子的容忍度更高。
    • 链表因为要存储指针,所以对于比较小的对象存储,是比较消耗内存的,还有可能会让内存消耗翻倍。而且,因为链表的结点是零散分布在内存中的,不是连续的,所以对CPU缓存是不友好的。当然,如果我们存储的是大对象,那链表中指针的内存消耗在大对象面前就可以忽略了。
    • 实际上,我们对链表表稍加改造,就可以实现一个更加高效的散列表。那就是,我们装饰链表法中的链表改造为其他高效的动态数据结构,比如跳表、红黑权。这样,即便出现散列冲突,极端情况下,的有的数据都散列到同一个桶内,寻最终退化成的散列表的查询时间也不过是O(logn)。这样也就有效避免了前面讲到的散列碰撞攻击。
    • 所以,基于链表的散列冲突处理方法比较适合存储大对象,大数据量的散列表,而且,比起开放寻址法,它更加灵活,支持更多的优化策略,比如红黑权代替链表。

06.工业级散列表案例

  • 刚刚讲了实现一个工业级散列表需要涉及的一些关键技术,现在,我就拿一个具体的例子,java 中的 HashMap 这样一个工业级的散列表,来具体看下,这些技术是怎么应用的。
  • 1、初始大小
    • HashMap 默认的初始大小是16,当然这个默认值是可以设置的,如果事先知道大概的数据量有多大,可以通过修改默认初始大小,减少动态扩容的次数,这样会大大提高HashMap的性能。
  • 2、装载因子和动态扩容
    • 最大装载因子默认是0.75,当HashMap 中的元素个数超过0.75*capacity( capacity 表示散列表的容量)的时候,就会启动扩容,每次扩容都会扩容为原来两倍的大小。
  • 3、散列冲突解决方法
    • HashMap 底层采用链表法解决冲突。即便负载因子和散列函数设计得再合理,也免不了会出现拉链过长的情况,一旦出现拉链过长,则会严重影响HashMap的性能。
    • 于是,在JDK1.8版本中,为了对HashMap做进一步优化,我们引入红黑权。而当链表长度太长(默认超过8)时,链表就转换为红黑树。我们可以利用红黑树快速增删改查的特点,提高HashMap的性能。当红黑树结点个数少于8个的时候,又会将红黑树转化为链表。因为在数据量较小的情况下,红黑树要维护平衡,比起链表来,性能的优势并不明显。
  • 4、散列函数
    • 散列函数的设计并不复杂,追求的是简单高效,健在均匀。我把它摘抄出来,你可以看看。
    int hash(Object key) {
        int h = key.hashCode();
        return (h ^ (h >>> 16)) & (capitity -1); //capicity表示散列表的大小
    }
    • 其中,hashCode()返回的是Java对象的 hash code 。比如 String 类型的对象的 hashCode()就是下面这样:
    public int hashCode(){
    	int var1 = this.hash;
    	if(var1 == 0 && this.value.length >0 ){
    		char[] var2 = this.value;
    		for(int var3 = 0; var3 < this.value.length; ++var3){
    			var = 31 * var1 + var2[var3];
    		}
    		this.hash = var1;
    	}
    	return var1;
    }

07.看思考题

  • 现在来分析一下开篇的问题:如何设计一个工业级的散列函数?如果这是一道面试题或者是摆在你面前的实际开发问题,你会从哪几个方面思考呢?
  • 会思考,何为一个工业级的散列表?工业级的散列表该具有哪些特性?结合已经学习过的散列知识,我觉得应该有这样几点要求:
    • 支持快速的查询、插入、删除操作;
    • 内存占用合理,不能浪费过多的内存空间;
    • 性能稳定,极端情况下,散列表的性能也不会退化到无法接受的情况。
  • **如何实现这样一个散列表呢?**根据前面讲过的知识,我会从这三个方面来考虑设计思路:
    • 设计一个合适的散列函数;
    • 定义装载因子阈值,并且设计动态扩容策略;
    • 选择合适的散列冲突解决方法。

08.内容小结

  • 上一节的内容比较偏理论,今天的内容侧重实战。我主要讲了如何设计一个工业级的散列表,以及如何应对各种异常情况,防止在极端情况下,散列表的性能退化过于严重。我分了三部分来讲解这些内容,分别是:如何设计散列函数,如何根据装载因子动态扩容,以及如何选择散列表冲突解决方法。
  • 关于散列函数的设计,我们要尽可能让散列后的值随机且均匀分布,这样会尽可能地减少散列冲突,即便冲突之后,分配到每个槽内的数据也比较均匀。除此之外,散列函数的设计也不能太复杂,太复杂就会太耗时间,也会影响散列表的性能。
  • 关于散列冲突解决方法的选择,我对比了开放寻址法和链表法两种方法的优劣和适应的场景。大部分情况下,链表法更加普适。而且,我们还可以通过将链表法中的链表改造成其他动态查找数据结构,比如红黑树,来避免散列表时间复杂度退化成O(n),抵御散列碰撞攻击。但是,对于小规模数据、装载因子不高的散列表,比较适合用开放寻址法。
  • 对于动态散列表来说,不管我们如何设计散列函数,选择什么样的散列冲突解决方法。随着数据的不断增加,散列表总会出现装载因子过高的情况。这个时候,我们就需要动态扩容。

01.先看一个案例

  • 一个excel文件的加密密码忘记了,是个六位全数字密码。我想打开这个文件,没有办法我分析了下可以用全枚举攻击强力破解它。破解时发现,当全枚举6位全数字密码时,程序跑飞,跑了一个小时没有结果。我就试5位密码结果几十秒就尝试完毕。看来6位密码破解受到计算机cpu等硬件和操作系统限制,难以实现。我就换一种想法,把六位全数字密码(10的6次方六位密码)分成10个密码本,一次破解一个密码本的秘密。破解10次就全部枚举完毕了,也要不了十几分钟,若运气好破解几千个就试出来了。终于这个文件的秘密被我采用字典式尝试出来了。

02.为何银行密码不易破解

  • 用银行卡的密码没有输入次数限制,知道用户名,破解密码也就十多分钟的事情,甚至优化后只需要几分钟。若你的密码是数字和大小写字母的混合(不包括标点符号),若密码长度达到32位,那是32的62次方个密码,理论上这种密码是没有破解的可能。
  • 但是这种原始方案的缺点也很显然,那就是这种密码太复杂太长,用户记不住,其次有被别人截获密码的可能,毕竟现在密码也需要在网上传输的。所以就产生了加密协议。加密协议的作用就是,用户如何一个简单密码,进行加密后就变成32位,64位,128位等密码了,然后在网上传输就安全多了,一般这种加密密码和时间戳也正相关。你截获了也用处不大,就是你把时间参数传递过去了,服务器也和本地时间比对的,超过3分钟他们就认为是非法消息,它是不断变化的,破解很困难。
  • 很多网站都有输入次数限制,所以对很多网站的密码破解都集中在加密算法上,很少进行字典式攻击了,当然黑客找到网站的漏洞,绕过次数限制,也会进行字典式轰炸。

03.一个优惠券的漏洞

  • 遇到一个离奇的事情。一个用户买了优惠券,使用时提示优惠券已经被使用过。但是用户刚刮开涂层,怎么就不能用了呢?原来优惠券是4位纯数字,发放的优惠券比较多,有用户买了优惠券使用了后,输入相连的号码,竟然有很大概率就能试出来有用的优惠券号码,并且客户端没有尝试次数的限制。这个就是现实的字典式攻击的例子。4位全数字密码也就9999个密码,他有多危险知道了吧!

04.md5是高级加密吗

  • md5是编码算法,别把它当成很高级的加密算法。网上就有字典式破解md5编码的网站,密码长不一定安全,那要看你的编码算法怎么样,最好对密码验证次数进行限时间段限制,时间正相关,风控管理(同一用户或同一ip进行短期大批量操作监控),密码设置为6位及以上数字和字母混合,那样能大大减少被破解的可能性。

参考博客

  • MD算法破解
    • https://blog.csdn.net/wufaliang003/article/details/79794982

参考文献

  • 极客时间,王争大神,数据结构和算法之美

01.关于博客汇总链接

  • 1.技术博客汇总
  • 2.开源项目汇总
  • 3.生活博客汇总
  • 4.喜马拉雅音频汇总
  • 5.其他汇总

02.关于我的博客

  • github:https://github.com/yangchong211
  • 知乎:https://www.zhihu.com/people/yczbj/activities
  • 简书:http://www.jianshu.com/u/b7b2c6ed9284
  • csdn:http://my.csdn.net/m0_37700275
  • 喜马拉雅听书:http://www.ximalaya.com/zhubo/71989305/
  • 开源中国:https://my.oschina.net/zbj1618/blog
  • 泡在网上的日子:http://www.jcodecraeer.com/member/content_list.php?channelid=1
  • 邮箱:yangchong211@163.com
  • 阿里云博客:https://yq.aliyun.com/users/article?spm=5176.100- 239.headeruserinfo.3.dT4bcV
  • segmentfault头条:https://segmentfault.com/u/xiangjianyu/articles
  • 掘金:https://juejin.im/user/5939433efe88c2006afa0c6e
贡献者: yangchong211