青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

無我

讓內心永遠燃燒著偉大的光明的精神之火!
靈活的思考,嚴謹的實現
豪邁的氣魄、頑強的意志和周全的思考

剖析eSNACC的hash函數

我們前面已經寫了一篇文章剖析eSNACC哈希結構的設計和實現 剖析eSNACC哈希結構的設計和實現 ,而本篇我們專門剖析eSNACC中的hash函數。

我們先看看eSNACC求hash函數的實現:

typedef unsigned int Hash;
/*
 *
 * From sdbm, an ndbm work-alike hashed database library
 * Author: oz@nexus.yorku.ca
 * Status: public domain.
 *
 * polynomial conversion ignoring overflows
 * [this seems to work remarkably well, in fact better
 * then the ndbm hash function. Replace at your own risk]
 * use: 65599   nice.
 *      65587   even better.
 *
 * [In one experiment, this function hashed 84165 symbols (English words
 * plus symbol table values) with no collisions. -bjb]
 *
 
*/


Hash    MakeHash (
const char *str, size_t len)
{
    register Hash n 
= 0;

#define HASHC   n = *str++ + 65587 * n

    
if (len > 0)
    
{
        
int loop;
        loop 
= (len + 8 - 1>> 3;
        
switch (len & (8 - 1))
    
{
          
case 0:          

            do
        
{
                HASHC;
              
case 7: HASHC;
              
case 6: HASHC;
              
case 5: HASHC;
              
case 4: HASHC;
              
case 3: HASHC;
              
case 2: HASHC;
              
case 1: HASHC;
        }
 while (--loop);
    }

    }

    
return n;
}

 看到這個代碼,我想先請問讀者您有什么想法?如果您沒來得及總結,可以參考我下面給的選項:

1、一定是本人粗心大意,copy代碼時弄錯了。

2、寫這個代碼的人是不是臨時工?基本的語法都沒過關。

3、大師,不愧為大師!

4、雕蟲小巧,小菜一碟,一目了然嘛~

 

=============================Please give your answers first=================

 

好了,我對您的選項作答:

1、我保證我做事一絲不茍、嚴謹踏實。此代碼是一字一句來自于源碼(除了空格的調整),如假包換、假一罰十!

2、我不能確定他是不是臨時工,但是我保證語法完全正確。

3、maybe...

4、那我只能說我太佩服您了!因為至少我是第一次看到case套在do-while里面的寫法。對您這樣的高人前來,有失遠迎,萬請恕罪!

 

==================================================================

好了,作為技術研究,我們下面就要深入剖析這個hash函數了。

首先,為了清楚,我把頭文件中的typedef也移到了代碼前。eSNACC的hash值用unsigned int表示。

然后,在說明代碼邏輯之前,我們不妨先看看函數的注釋:作者說本函數來自于sdbm,其中的修改就是把原來的magic number 65599改為了65587,因為他說明:65599 nice ,65587   even better.哪個魔數是better,我們就不在本文討論了。我們只研究算法。

既然是來自于sdbm,那么sdbm是什么呢?我的另一篇博客專門轉錄了這些信息,有興趣可以參見hash函數——djb2、sdbm、lose lose

這里只簡單介紹一下sdbm:

sdbm哈希函數的算法:對一個字符串str,分別求出hash(i) = hash(i - 1) * 65599 + str[i],hash值就是所有hash(i)的和。其實現為:

  static unsigned long    sdbm( unsigned char *str)
  {
        unsigned long hash = 0;
        int c;

        while (c = *str++)
            hash = c + (hash << 6) + (hash << 16) - hash;

        return hash;
    }

 

然后我們再回到eSNACC中的MakeHash函數,看那種讓人崩潰的代碼是不是就是這個算法:

先看這個宏#define HASHC   n = *str++ + 65587 * n,這好像表達的就是hash(i) = hash(i - 1) * 65599 + str[i],只是65599->65587.這個讓我們很滿意,那么后面的switch等是不是就是一個遍歷呢?

loop = (len + 8 - 1) >> 3   ==> loop=(len+8-1)/8 ==> loop=(len-1)/8 + 1  :其實算出來的loop就是len除以8的值,如果有余數,那值加1。

len & (8 - 1) :這里算出來的恰恰就是len%8的值。

我們再仔細分析switch-case和do-while結構:

其算法是這樣的:比如len=28,

1.計算loop=4.

2.算出len%8的值,然后先執行該值次數的HASHC;本例中len%8=4,那么在switch時會到case 4:然后依次執行了4次HASHC;

3.進while,--loop,這樣這個do-while還會執行3次,每次HASHC會運行8次。

所以,最終就是3*8 + 4 =28,也就是說這與上面的sdbm的算法時完全一樣的!只是魔數變了,然后寫法上有些匪夷所思了。

 

/***********************************休息一下**************************************

好了,對MakeHash函數的分析是完成了,但是我覺得很有必要來思考一下這個函數的效率。

我們設計hash函數,一方面希望使函數產生的哈希值盡量分散減少沖突,另一方面就是希望保證這個函數效率很高,因為每次加入值,查找值時都要計算hash,如果函數效率不高,對整個系統也會是一種負擔。

結合這兩個因素,第一、我們發現作者將65599改成65587,他說明這樣生成hash的效果更好。但是第二條就實在無法讓人樂觀了!我們看到hash函數——djb2、sdbm、lose lose 中給的hash函數都為了提高性能都已經把對魔數的運算改成了移位操作,但是在MakeHash中,就是死硬的乘法!對字符串的每一個字節算一次乘法!說實話,想起如果字符串比較長,我真有點毛骨悚然...

 

所以,就讓我們就仿照sdbm中的實現,把乘法優化掉來做一個MakeHash的優化版吧:

Hash MakeHashOpt (const char *str, size_t len)
{
    register Hash n 
= 0;

#define HASHCOPT   n = *str++ + (n << 16) + (n << 5) +  (n << 4) + (n << 1) + n

    
if (len > 0)
    
{
        
int loop;
        loop 
= (len + 8 - 1>> 3;
        
switch (len & (8 - 1))
        
{
        
case 0:        
            
do
            
{
                HASHCOPT;
        
case 7: HASHCOPT;
        
case 6: HASHCOPT;
        
case 5: HASHCOPT;
        
case 4: HASHCOPT;
        
case 3: HASHCOPT;
        
case 2: HASHCOPT;
        
case 1: HASHCOPT;
            }
 while (--loop);
        }

    }

    
return n;
}

 

最后,讓我們來分析一下作者設計成這種看起來很別扭的實現形式的目的吧。

正如前面的分析,這樣做事先求出len對8的倍數,然后除了需要對余數做switch-case判斷來宏調用1-8次以為,其他的就是每次固定做8次宏調用。我認為:他這樣做,就是希望減少用while或for遍歷串的判斷次數,因為那樣要判斷len+1次?,F在就只要len/8 + 1 + len%8次了。

 

好了,對eSNACC的hash函數剖析就到此了。

 

 

posted on 2012-04-26 15:37 Tim 閱讀(1704) 評論(2)  編輯 收藏 引用 所屬分類: eSNACC學習

評論

# re: 剖析eSNACC的hash函數 2012-04-26 15:55 Tim

呵呵,謝謝支持~@嵌入式培訓
  回復  更多評論   

# re: 剖析eSNACC的hash函數[未登錄] 2012-04-27 09:29 Tina

不懂技術,還是支持一下!加油!  回復  更多評論   

<2009年9月>
303112345
6789101112
13141516171819
20212223242526
27282930123
45678910

導航

統計

公告

本博客原創文章,歡迎轉載和交流。不過請注明以下信息:
作者:TimWu
郵箱:timfly@yeah.net
來源:www.shnenglu.com/Tim
感謝您對我的支持!

留言簿(9)

隨筆分類(173)

IT

Life

搜索

積分與排名

最新隨筆

最新評論

閱讀排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            亚洲精品久久久久久久久久久久| 久久九九免费视频| 欧美乱大交xxxxx| 一区二区三区高清| 欧美日韩亚洲一区二区三区| 一本到高清视频免费精品| 99国产精品私拍| 国产精品久久久久久久久久久久久| 亚洲素人一区二区| 亚洲视频一区在线| 国产日韩欧美黄色| 蜜桃av综合| 欧美人牲a欧美精品| 亚洲一区欧美激情| 欧美中文字幕久久| 亚洲国产精品第一区二区| 亚洲欧洲综合| 国产精品男人爽免费视频1 | 欧美激情无毛| 羞羞视频在线观看欧美| 久久久精品网| 亚洲一区二区三区精品在线观看| 亚洲视频在线一区观看| 狠狠做深爱婷婷久久综合一区| 欧美激情一区二区三区蜜桃视频 | 亚洲韩国青草视频| 亚洲美女毛片| 国产亚洲精久久久久久| 亚洲国产专区校园欧美| 国产精品午夜在线观看| 亚洲电影在线免费观看| 国产欧美视频一区二区| 亚洲国产乱码最新视频| 国产欧美视频一区二区三区| 亚洲精品一区二区三区蜜桃久 | 欧美高清日韩| 久久一区二区三区av| 欧美性大战xxxxx久久久| 男人的天堂成人在线| 国产精品久久午夜| 最近看过的日韩成人| 激情久久综艺| 亚洲一区免费在线观看| 一区二区三区欧美在线| 美女图片一区二区| 久久精品国产77777蜜臀| 亚洲高清视频在线| 黄色av成人| 午夜免费日韩视频| 在线一区二区三区做爰视频网站| 久久综合给合| 久久久精品国产免费观看同学| 欧美午夜不卡在线观看免费 | 日韩视频在线一区二区| 久久理论片午夜琪琪电影网| 欧美一区二区免费观在线| 欧美区日韩区| 亚洲欧洲精品成人久久奇米网| 黄色免费成人| 久久爱另类一区二区小说| 午夜精品区一区二区三| 久久精品一级爱片| 国产啪精品视频| 亚洲伊人网站| 欧美一区二区在线免费观看| 欧美午夜剧场| 亚洲少妇中出一区| 亚洲欧美不卡| 国产精品乱看| 午夜影视日本亚洲欧洲精品| 欧美在线观看www| 国产一区二区精品| 久久先锋影音| 亚洲丁香婷深爱综合| 亚洲精品日产精品乱码不卡| 欧美激情一区二区三区在线| 亚洲区中文字幕| 9久草视频在线视频精品| 欧美日韩免费观看一区三区| 夜夜嗨av一区二区三区网页| 性欧美大战久久久久久久免费观看 | 欧美激情一区二区三区在线视频观看 | 亚洲欧美激情视频| 久久久久久久久岛国免费| 国产在线不卡| 免费亚洲电影在线| 亚洲乱码国产乱码精品精98午夜| 99精品欧美一区| 国产精品观看| 久久精品亚洲一区二区三区浴池| 欧美freesex8一10精品| 一区二区高清视频| 国产一区二区三区自拍| 欧美粗暴jizz性欧美20| 中文国产成人精品| 久久婷婷综合激情| 一区二区免费在线播放| 国产伦精品一区二区| 久久综合给合久久狠狠狠97色69| 亚洲伦理在线免费看| 欧美一区二区三区视频免费| 韩国视频理论视频久久| 欧美日本精品一区二区三区| 午夜欧美精品| 亚洲人成小说网站色在线| 欧美一级一区| 亚洲精品激情| 国产综合欧美| 国产精品wwwwww| 久久综合狠狠| 亚洲影音先锋| 在线综合欧美| 伊人久久大香线蕉av超碰演员| 欧美女同视频| 久久久久一区二区三区| 亚洲图片欧美一区| 91久久久久久久久| 久久精品主播| 亚洲欧美国内爽妇网| 亚洲黄色av| 国语自产精品视频在线看一大j8 | 一区二区三区视频观看| 亚洲第一在线综合网站| 亚洲福利视频三区| 久久久久综合网| 亚洲欧美激情四射在线日| 亚洲区中文字幕| 一区二区在线视频| 国产亚洲欧美一区二区| 国产精品美女诱惑| 欧美日韩aaaaa| 欧美va日韩va| 免费黄网站欧美| 久久国产视频网站| 午夜欧美精品| 亚洲欧美视频在线观看视频| 一本色道久久综合亚洲精品按摩| 亚洲国产裸拍裸体视频在线观看乱了| 久久亚洲精品伦理| 久久久噜久噜久久综合| 久久高清免费观看| 久久成人免费电影| 久久aⅴ国产紧身牛仔裤| 香蕉久久a毛片| 欧美在线|欧美| 亚洲欧美成人综合| 欧美在线播放| 久久女同互慰一区二区三区| 久久久久久9999| 久久国产精品亚洲va麻豆| 性欧美xxxx大乳国产app| 欧美在线视屏| 久久婷婷久久| 欧美国产成人精品| 亚洲精品小视频| 中日韩视频在线观看| 亚洲一区二区欧美日韩| 欧美影院在线播放| 久久精品国产免费| 久久综合伊人77777| 欧美激情一区三区| 国产精品v欧美精品∨日韩| 国产精品欧美经典| 黑人巨大精品欧美黑白配亚洲| 亚洲大黄网站| 夜色激情一区二区| 小辣椒精品导航| 免费在线欧美黄色| 亚洲九九精品| 欧美一区二区视频在线| 欧美3dxxxxhd| 国产精品美女| 在线成人黄色| 亚洲视频福利| 久久久亚洲一区| 亚洲激情校园春色| 亚洲欧美综合网| 欧美黄色小视频| 国产日韩欧美成人| 亚洲免费播放| 久久激情网站| 亚洲乱码国产乱码精品精| 欧美亚洲一区二区在线| 欧美日韩dvd在线观看| 狠狠色丁香久久综合频道| 99国产精品久久久| 久久婷婷蜜乳一本欲蜜臀| 亚洲一区二区三区四区在线观看 | 日韩视频在线免费观看| 先锋影音网一区二区| 欧美国产1区2区| 国产性天天综合网| 亚洲一二三区在线| 欧美高清你懂得| 性久久久久久久久久久久| 欧美区亚洲区| 91久久在线| 老牛嫩草一区二区三区日本 | 免费不卡视频| 先锋资源久久|