`
zhaohaolin
  • 浏览: 1016274 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

打造最快的Hash表[转]

阅读更多

打造最快的Hash表(暴雪用的MPQ文件)

 

最近在网上看到篇文章,一起拜一拜暴雪

先提一个简单的问题,如果有一个庞大的字符串数组,然后给你一个单独的字符串,让你从这个数组中查找是否有这个字符串并找到它,你会怎么做?

有一个方法最简单,老老实实从头查到尾,一个一个比较,直到找到为止,我想只要学过程序设计的人都能把这样一个程序作出来,但要是有程序员把这样的程序交给用户,我只能用无语来评价,或许它真的能工作,但...也只能如此了。

最合适的算法自然是使用HashTable(哈希表),先介绍介绍其中的基本知识,所谓Hash,一般是一个整数,通过某种算法,可以把一个字符串"压缩" 成一个整数,这个数称为Hash,当然,无论如何,一个32位整数是无法对应回一个字符串的,但在程序中,两个字符串计算出的Hash值相等的可能非常小,下面看看在MPQ中的Hash算法

unsigned long HashString(char *lpszFileName, unsigned long dwHashType)

 unsigned char *key = (unsigned char *)lpszFileName;
unsigned long seed1 = 0x7FED7FED, seed2 = 0xEEEEEEEE;
int ch;

while(*key != 0)

  ch = toupper(*key++);

seed1 = cryptTable[(dwHashType << 8) + ch] ^ (seed1 + seed2);
seed2 = ch + seed1 + seed2 + (seed2 << 5) + 3; 
 }
return seed1; 

Blizzard的这个算法是非常高效的,被称为"One-Way Hash",举个例子,字符串"unitneutralacritter.grp"通过这个算法得到的结果是0xA26067F3。
是不是把第一个算法改进一下,改成逐个比较字符串的Hash值就可以了呢,答案是,远远不够,要想得到最快的算法,就不能进行逐个的比较,通常是构造一个哈希表(Hash Table)来解决问题,哈希表是一个大数组,这个数组的容量根据程序的要求来定义,例如1024,每一个Hash值通过取模运算 (mod)对应到数组中的一个位置,这样,只要比较这个字符串的哈希值对应的位置又没有被占用,就可以得到最后的结果了,想想这是什么速度?是的,是最快的O(1),现在仔细看看这个算法吧
int GetHashTablePos(char *lpszString, SOMESTRUCTURE *lpTable, int nTableSize)

 int nHash = HashString(lpszString), nHashPos = nHash % nTableSize;

if (lpTable[nHashPos].bExists && !strcmp(lpTable[nHashPos].pString, lpszString)) 
  return nHashPos; 
 else 
  return -1; //Error value 

看到此,我想大家都在想一个很严重的问题:"如果两个字符串在哈希表中对应的位置相同怎么办?",毕竟一个数组容量是有限的,这种可能性很大。解决该问题的方法很多,我首先想到的就是用"链表",感谢大学里学的数据结构教会了这个百试百灵的法宝,我遇到的很多算法都可以转化成链表来解决,只要在哈希表的每个入口挂一个链表,保存所有对应的字符串就OK了。

事情到此似乎有了完美的结局,如果是把问题独自交给我解决,此时我可能就要开始定义数据结构然后写代码了。然而Blizzard的程序员使用的方法则是更精妙的方法。基本原理就是:他们在哈希表中不是用一个哈希值而是用三个哈希值来校验字符串。

中国有句古话"再一再二不能再三再四",看来Blizzard也深得此话的精髓,如果说两个不同的字符串经过一个哈希算法得到的入口点一致有可能,但用三个不同的哈希算法算出的入口点都一致,那几乎可以肯定是不可能的事了,这个几率是1:18889465931478580854784,大概是10的 22.3次方分之一,对一个游戏程序来说足够安全了。

现在再回到数据结构上,Blizzard使用的哈希表没有使用链表,而采用"顺延"的方式来解决问题,看看这个算法:
int GetHashTablePos(char *lpszString, MPQHASHTABLE *lpTable, int nTableSize)

 const int HASH_OFFSET = 0, HASH_A = 1, HASH_B = 2;
int nHash = HashString(lpszString, HASH_OFFSET);
int nHashA = HashString(lpszString, HASH_A);
int nHashB = HashString(lpszString, HASH_B);
int nHashStart = nHash % nTableSize, nHashPos = nHashStart;

while (lpTable[nHashPos].bExists)

  if (lpTable[nHashPos].nHashA == nHashA && lpTable[nHashPos].nHashB == nHashB) 
   return nHashPos; 
  else 
   nHashPos = (nHashPos + 1) % nTableSize;
  
  if (nHashPos == nHashStart) 
   break; 
 }

return -1; //Error value 

1. 计算出字符串的三个哈希值(一个用来确定位置,另外两个用来校验)
2. 察看哈希表中的这个位置
3. 哈希表中这个位置为空吗?如果为空,则肯定该字符串不存在,返回
4. 如果存在,则检查其他两个哈希值是否也匹配,如果匹配,则表示找到了该字符串,返回
5. 移到下一个位置,如果已经越界,则表示没有找到,返回
6. 看看是不是又回到了原来的位置,如果是,则返回没找到
7. 回到3

怎么样,很简单的算法吧,但确实是天才的idea, 其实最优秀的算法往往是简单有效的算法.

分享到:
评论

相关推荐

    打造最快的Hash表

    哈希表(Hash Table)是一种数据结构,它通过哈希函数将输入(通常是字符串)映射到一个固定大小的数组的索引上,使得数据的查找、插入和删除操作能够达到近乎常数时间的复杂度,即O(1)。在编程中,哈希表的高效性能...

    高手打造最快的Hash表源码(和Blizzard的对话)

    在《高手打造最快的Hash表源码(和Blizzard的对话)》这个主题中,我们可以期待深入探讨这些细节,学习如何在实际项目中优化Hash表的性能,以满足像Blizzard这样的顶级游戏公司的需求。这份源码可能会包含独特的设计...

    打造最快的Hash表(和Blizzard的对话)

    ### 打造最快的Hash表(和Blizzard的对话) #### Hash表基础知识与应用场景 本文将深入探讨如何构建高效的Hash表,并特别关注Blizzard在游戏开发过程中所采用的技术。Hash表是一种利用散列函数来实现快速查找的数据...

    打造最快xp系统的工具

    标题中的“打造最快XP系统的工具”指的是通过特定的软件或方法优化Windows XP操作系统,使其运行速度得到显著提升。XP系统是微软在2001年发布的一款经典操作系统,虽然已经停止官方支持,但在某些领域仍有用户在使用...

    从头到尾彻底解析Hash_表算法

    #### 第三部分:打造一个最快的哈希表算法 为了构建一个高性能的哈希表,需要综合考虑以下几个因素: - **哈希函数的选择**:选择合适的哈希函数是提高哈希表性能的关键。理想情况下,哈希函数应该尽可能均匀地...

    Hash 哈希学习的一些资料

    自己学习Hash时搜到的一些资料。包括rfc1321.txt MPQ Hash.txt Inside MoPaQ - Chapter 2 Fundamentals.mht 各种字符串HASH函数 打造最快的Hash表 & Times33 哈希算法(Hash Algorithm)初探 关于hash的一些问题等。

    从头到尾彻底解析hash

    #### 第三部分:打造最快的Hash表算法 为了构建高效的哈希表,需关注以下几点: - **散列函数优化**:设计或选用具有良好分布特性的散列函数,减少碰撞。 - **动态调整容量**:根据负载因子(填充率)自动调整散...

    转转如何打造AI工程架构体系.rar

    在IT行业中,构建高效、可扩展的AI工程架构体系对于任何企业来说都是至关重要的,尤其对于像“转转”这样的平台。本案例将探讨“转转”如何通过一系列技术和策略来构建其AI工程架构,旨在提供一个技术实践的参考样本...

    浩辰CAD创新技术打造全球最快“引擎”.pdf

    浩辰CAD创新技术打造全球最快“引擎”.pdf

    转转-孙玄-如何快速突破个人成长非连续性,不断打造成长第二曲线.pdf

    转转-孙玄-如何快速突破个人成长非连续性,不断打造成长第二曲线

    打造竞争优势的战略转型概述.doc

    打造竞争优势的战略转型概述.doc

    快表软件正式企业版V5.2.1官方免费安装版

    快表软件正式企业版是基于快表软件制作的正式企业版本,该款软件专为企业用户量身打造,支持Windows以及Mac系统,包含了模块功能、任务代办、模版权限、权限复制、用户组权限、数据备份与恢复、多维分析、智能图表、...

    转转如何打造AI工程架构体系.pdf

    转转如何打造AI工程架构体系.pdf

    快表软件国际版V4.0.0官方免费安装版

    快表软件国际版是专为快表软件用户制作的国际版本,该款软件比起企业版和免费版,支持多语言环境,专为全球业务打造,让语言变得毫无障碍,包含了免费版所有功能的基础上,还拥有诸多高级功能,包含了超强自定义SQL...

    快速打造你的赚钱机器

    打造你的赚钱机器,轻松赚钱潇洒生活 打造你的赚钱机器,轻松赚钱潇洒生活 打造你的赚钱机器,轻松赚钱潇洒生活 打造你的赚钱机器,轻松赚钱潇洒生活

    打造国内首个非转基因大豆食品科技产业园宣传册.docx

    ### 打造国内首个非转基因大豆食品科技产业园关键知识点 #### 一、项目背景与意义 - **项目名称**:打造国内首个非转基因大豆食品科技产业园 - **目标定位**:建立中国首个专注于非转基因大豆食品的研发、生产和...

    Vc++ 打造局域网聊天室

    Vc++ 打造局域网聊天室(1)---编程环境及准备知识 http://dx.3800hk.com/donghua/g/27288.html Vc++ 打造局域网聊天室(2)---聊天室界面的制作及初始化 http://www.3800hk.com/donghua/g/27317.html Vc++ 打造局域网...

    nginx_upstream_hash-0.3.2.tar.gz

    《深入解析nginx_upstream_hash模块:打造高效负载均衡策略》 在互联网服务中,nginx作为一款高性能的HTTP和反向代理服务器,广泛应用于各种复杂的网络架构中。为了满足动态负载均衡的需求,nginx提供了丰富的...

    打造国内首个非转基因大豆食品科技产业园宣传册-3页.pdf

    【标题】和【描述】提及的是打造国内首个非转基因大豆食品科技产业园的计划,这个产业园位于黑龙江省嫩江县,一个在大豆种植方面具有显著优势的地区。以下是对该知识点的详细阐述: 嫩江县作为大豆主产区,具备...

Global site tag (gtag.js) - Google Analytics