• <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>

            浪跡天涯

            唯有努力...
            努力....再努力...

            CMap詳解

            如何聲明CMap
            許多人對Cmap的聲明模式CMap<KEY,ARG_KEY,VALUE,ARG_VALUE>感到迷惑,為什么不用CMap<KEY,VALUE>呢?實際上,CMap中的的數據最終會是CPair,而CPair內部是(KEY,VALUE)。因此,CMap其實存儲的是KEY,而非ARG_KEY。然而,如果你查看MFC的源代碼,幾乎CMap所有的內部參數傳遞都是訪問ARG_KEY和ARG_VALUE,因此,使用KEY&來代替ARG_KEY似乎是正確的,除了在這些情況下:
            1 應用簡單的數據類型,如int ,char用值傳遞與參數傳遞沒有什么不同
            2 如果用CString作為KEY,你應該用LPCTSTR   做ARG_KEY而非CString&,接下來我門會討論原因。

            如何讓CMap類為自己工作
            好的,就象我前面說過的,CMap是一個哈西表,一個哈西表要有“哈西值“——一個UINT類型,用哈西值作為在哈西表中的序數。如果有更多的相同的關鍵字,他們會組成一個鏈表。因此,你應該首先構造哈西函數。CMap類會調用摸板函數HashKey()來構造哈西函數。缺省應用和特別版的LPCSTR和LPCWSTR如下:

            // inside <afxtemp.h>
            template<class ARG_KEY>
            AFX_INLINE UINT AFXAPI HashKey(ARG_KEY key)
            {
                // default identity hash - works for most primitive values
                return (DWORD)(((DWORD_PTR)key)>>4);
            }// inside <strcore.cpp>
            // specialized implementation for LPCWSTR
            #if _MSC_VER >= 1100
            template<> UINT AFXAPI HashKey<LPCWSTR> (LPCWSTR key)
            #else
            UINT AFXAPI HashKey(LPCWSTR key)
            #endif
            {
                UINT nHash = 0;
                while (*key)
                    nHash = (nHash<<5) + nHash + *key++;
                return nHash;
            }// specialized implementation for LPCSTR
            #if _MSC_VER >= 1100
            template<> UINT AFXAPI HashKey<LPCSTR> (LPCSTR key)
            #else
            UINT AFXAPI HashKey(LPCSTR key)
            #endif
            {
                UINT nHash = 0;
                while (*key)
                    nHash = (nHash<<5) + nHash + *key++;
                return nHash;
            }
            正如你所見到的,缺省行為是“假定“關鍵字是一個指針,并且轉變成DWORD類型,這就是為什么會出現“error C2440:’type cast’:cannot convert from ‘ClassXXX’to ‘DWORD_PTR’”如果你不提供一個特別的HashKey()函數給你的類就會出現上述情況。并且由于MFC僅僅提供了特殊的工具LPCSTR和LPCWSTR,卻沒有提供CStringA或CStringW,如果你想要在CMap中用CString,就必須聲明CMap<CString ,LPCSTR….>,OK,現在你知道怎么計算CMap的哈西值了,但是因為一個關鍵字可能對應多個哈西值,CMap就需要找遍整個鏈表來找到正確的“摸板”,不僅用同樣的“哈西值”。當CMap不匹配時,就會訪問CompareElements(),另一個摸板方程。// inside <afxtemp.h>
            // noted: when called from CMap,
            //        TYPE=KEY, ARG_TYPE=ARG_TYPE
            // and note pElement1 is TYPE*, not TYPE
            template<class TYPE, class ARG_TYPE>
            BOOL AFXAPI CompareElements(const TYPE* pElement1,
                                        const ARG_TYPE* pElement2)
            {
                ASSERT(AfxIsValidAddress(pElement1,
                       sizeof(TYPE), FALSE));
                ASSERT(AfxIsValidAddress(pElement2,
                       sizeof(ARG_TYPE), FALSE));    // for CMap<CString, LPCTSTR...>
                // we are comparing CString == LPCTSTR
                return *pElement1 == *pElement2;
            }
            因此,如果你想在自己的類中用CMap,你不得不重寫HashKey()和CompareElements()
            結束語
            1 CMap是一個哈西表,而STL::map是一個樹表,對他們做比較是沒有意義的。但是,如果你你要重新找到有序的關鍵字,你就得使用STL::map
            2 HashKey()的設計是高效的。你應該提供一個較少沖突的HashKey(),并且容易計算。你要記注,對于有些類來說,這不容易。
            3 當用Cmap(或STL::hash_map),要注意哈西表的大小。

            附能用于CString的CMap重寫的HashKey()和CompareElements()
            using namespace std;
            template<>
            UINT AFXAPI HashKey<CString*> (CString* key)
            {
             return (NULL == key) ? 0 : HashKey((LPCTSTR)(*key));
            }

            // I don't know why, but CompareElements can't work with CString*
            // have to define this
            typedef CString* LPCString;

            template<>
            BOOL AFXAPI CompareElements<LPCString, LPCString> (const LPCString* pElement1,
                           const LPCString* pElement2)
            {
             if ( *pElement1 == *pElement2 ) {
              // true even if pE1==pE2==NULL
              return true;
             } else if ( NULL != *pElement1 && NULL != *pElement2 ) {
              // both are not NULL
              return **pElement1 == **pElement2;
             } else {
              // either one is NULL
              return false;
             }
            }:

            posted on 2008-01-24 15:41 浪跡天涯 閱讀(4283) 評論(2)  編輯 收藏 引用 所屬分類: C++

            評論

            # re: CMap詳解 2008-01-24 15:47 浪跡天涯

            以下是實現忽略大小寫的HashKey函數以及KeyCompare函數:
            // 實現忽略大小寫的
            template<>
            inline bool HS_HashKey<char*>::KeyCompare( char* const&key1, char* const&key2)
            {
            return stricmp(key1, key2) == 0;
            }

            template<>
            inline unsigned int HS_HashKey<char*>::KeyHash( char* const&lkey)
            {
            unsigned int nHash = 0;
            const char* key = lkey;
            while (*key)
            {
            if(*key >= 'A' && *key <= 'Z')
            {
            nHash = (nHash<<5) + nHash + *key++;
            nHash += ('a' - 'A');
            }
            else
            nHash = (nHash<<5) + nHash + *key++;
            }
            return nHash;
            }
              回復  更多評論   

            # re: CMap詳解 2008-06-06 17:21 11

            不知所云  回復  更多評論   

            <2008年1月>
            303112345
            6789101112
            13141516171819
            20212223242526
            272829303112
            3456789

            導航

            統計

            常用鏈接

            留言簿(22)

            隨筆分類(30)

            隨筆檔案(29)

            文章分類

            搜索

            積分與排名

            最新評論

            閱讀排行榜

            評論排行榜

            久久99精品国产麻豆不卡| 99久久成人国产精品免费| 99久久99久久精品国产片果冻| 99久久夜色精品国产网站| 日本亚洲色大成网站WWW久久| 久久久久久久91精品免费观看| 久久久精品人妻一区二区三区蜜桃| 亚洲精品乱码久久久久66| 国产精品久久久久久影院 | 91精品国产综合久久精品| 久久国产成人午夜aⅴ影院| 77777亚洲午夜久久多人| 久久国产精品成人免费 | 国产AⅤ精品一区二区三区久久| 色天使久久综合网天天| 精品久久久久香蕉网| 日本加勒比久久精品| 丁香五月综合久久激情| 久久精品麻豆日日躁夜夜躁| 最新久久免费视频| 国产精品日韩欧美久久综合| 久久久噜噜噜久久中文福利| 国产精品乱码久久久久久软件 | 久久综合综合久久狠狠狠97色88| 久久精品一本到99热免费| 国产精品热久久无码av| 久久精品嫩草影院| 欧美一区二区三区久久综合 | 久久国产成人精品国产成人亚洲| 伊人久久大香线焦AV综合影院 | 精品免费tv久久久久久久| 无码人妻久久一区二区三区免费| 一级女性全黄久久生活片免费 | 国产毛片久久久久久国产毛片| 久久久无码精品亚洲日韩蜜臀浪潮 | 精品伊人久久大线蕉色首页| 91精品国产综合久久四虎久久无码一级| 久久精品国产色蜜蜜麻豆| 久久久国产99久久国产一| 午夜精品久久久久9999高清| 久久香蕉国产线看观看猫咪?v|