祝愿你们不要像菜菜这般苦逼,年中奖大大滴
在没有年终奖的日子里,工做依然还要继续.....一张冰与火的图尽显无奈
![]()
还记得菜菜不久以前设计的用户空间吗?没看过的同窗请进传送门=》设计高性能访客记录系统算法
还记得遗留的什么问题吗?菜菜来重复一下,在用户访问记录的缓存中怎么来判断是否有当前用户的记录呢?链表虽然是咱们这个业务场景最主要的数据结构,但并非当前这个问题最好的解决方案,因此咱们须要一种能快速访问元素的数据结构来解决这个问题?那就是今天咱们要谈一谈的 散列表c#
散列表(Hash table,也叫哈希表),是根据关键码值(Key value)而直接进行访问的数据结构。也就是说,它经过把关键码值映射到表中一个位置来访问记录,以加快查找的速度。这个映射函数叫作散列函数,存放记录的数组叫作散列表。
散列表其实能够约等于咱们常说的Key-Value形式。
散列表用的是数组支持按照下标随机访问数据的特性,因此散列表其实就是数组的一种扩展,由数组演化而来。能够说,若是没有数组,就没有散列表。为何要用数组呢?由于数组按照下标来访问元素的时间复杂度为O(1),不明白的同窗能够参考菜菜之前的关于数组的文章。既然要按照数组的下标来访问元素,必然也必须考虑怎么样才能把Key转化为下标。这就是接下来要谈一谈的散列函数。
散列函数通俗来说就是把一个Key转化为数组下标的黑盒。散列函数在散列表中起着很是关键的做用。
散列函数,顾名思义,它是一个函数。咱们能够把它定义成hash(key),其中 key 表示元素的键值,hash(key) 的值表示通过散列函数计算获得的散列值。
那一个散列函数有哪些要求呢?数组
简单说一下以上三点,第一点:由于散列值其实就是数组的下标,因此必须是非负整数(>=0),第二点:同一个key计算的散列值必须相同。
重点说一下第三点,其实第三点只是理论上的,咱们想象着不一样的Key获得的散列值应该不一样,可是事实上,这一点很难作到。咱们能够反证一下,若是这个公式成立,我计算无限个Key的散列值,那散列表底层的数组必须作到无限大才行。像业界比较著名的MD五、SHA等哈希算法,也没法彻底避免这样的冲突。固然若是底层的数组越小,这种冲突的概率就越大。因此一个完美的散列函数实际上是不存在的,即使存在,付出的时间成本,人力成本可能超乎想象。缓存
既然再好的散列函数都没法避免散列冲突,那咱们就必须寻找其余途径来解决这个问题。网络
若是遇到冲突的时候怎么办呢?方法之一是在冲突的位置开始找数组中空余的空间,找到空余的空间而后插入。就像你去商店买东西,发现东西卖光了,怎么办呢?找下一家有东西卖的商家买呗。
无论采用哪一种探测方法,当散列表中空闲位置很少的时候,散列冲突的几率就会大大提升。为了尽量保证散列表的操做效率,通常状况下,咱们会尽量保证散列表中有必定比例的空闲槽位。咱们用装载因子(load factor)来表示空位的多少。数据结构
散列表的装载因子 = 填入表中的元素个数 / 散列表的长度
装载因子越大,说明空闲位置越少,冲突越多,散列表的性能会降低. 假设散列函数为 f=(key%1000),以下图所示多线程
拉链法属于一种最经常使用的解决散列值冲突的方式。基本思想是数组的每一个元素指向一个链表,当散列值冲突的时候,在链表的末尾增长新元素。查找的时候同理,根据散列值定位到数组位置以后,而后沿着链表查找元素。若是散列函数设计的很是糟糕的话,相同的散列值很是多的话,散列表元素的查找会退化成链表查找,时间复杂度退化成O(n)框架
这种方式本质上是计算屡次散列值,那就必然须要多个散列函数,在产生冲突时再使用另外一个散列函数计算散列值,直到冲突再也不发生,这种方法不易产生“汇集”,但增长了计算时间。分布式
至于这种方案网络上介绍的比较少,通常应用的也比较少。能够这样理解:散列值冲突的元素放到另外的容器中,固然容器的选择有多是数组,有多是链表甚至队列均可以。可是不管是什么,想要保证散列表的优势仍是须要慎重考虑这个容器的选择。函数
有几个地方菜菜须要在强调一下:
用户访问记录的实体
class UserViewInfo { //用户ID public int UserId { get; set; } //访问时间,utc时间戳 public int Time { get; set; } //用户姓名 public string UserName { get; set; } }
用户空间添加访问记录的代码
class UserSpace { //缓存的最大数量 const int CacheLimit = 1000; //这里用双向链表来缓存用户空间的访问记录 LinkedList<UserViewInfo> cacheUserViewInfo = new LinkedList<UserViewInfo>(); //这里用哈希表的变种Dictionary来存储访问记录,实现快速访问,同时设置容量大于缓存的数量限制,减少哈希冲突 Dictionary<int, UserViewInfo> dicUserView = new Dictionary<int, UserViewInfo>(1250); //添加用户的访问记录 public void AddUserView(UserViewInfo uv) { //首先查找缓存列表中是否存在,利用hashtable来实现快速查找 if (dicUserView.TryGetValue(uv.UserId, out UserViewInfo currentUserView)) { //若是存在,则把该用户访问记录从缓存当前位置移除,添加到头位置 cacheUserViewInfo.Remove(currentUserView); cacheUserViewInfo.AddFirst(currentUserView); } else { //若是不存在,则添加到缓存头部 并添加到哈希表中 cacheUserViewInfo.AddFirst(uv); dicUserView.Add(uv.UserId, uv); } //这里每次都判断一下缓存是否超过限制 if (cacheUserViewInfo.Count > CacheLimit) { //移除缓存最后一个元素,并从hashtable中删除,理论上来讲,dictionary的内部会两个指针指向首元素和尾元素,因此查找这两个元素的时间复杂度为O(1) var lastItem = cacheUserViewInfo.Last.Value; dicUserView.Remove(lastItem.UserId); cacheUserViewInfo.RemoveLast(); } } }
添加关注,查看更精美版本,收获更多精彩