• <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>
            <2011年12月>
            27282930123
            45678910
            11121314151617
            18192021222324
            25262728293031
            1234567

            統(tǒng)計(jì)

            • 隨筆 - 24
            • 文章 - 0
            • 評(píng)論 - 17
            • 引用 - 0

            常用鏈接

            留言簿(4)

            隨筆分類

            隨筆檔案

            相冊(cè)

            搜索

            •  

            最新評(píng)論

            閱讀排行榜

            評(píng)論排行榜

            BSTR CComBSTR及ATL字符串轉(zhuǎn)換宏
            一 BSTR及CComBSTR

            MSDN文檔中說(shuō),BSTR是四字節(jié)長(zhǎng)度前綴的,NULL結(jié)尾的寬字符串,稱之為VB字符串或BINARY字符串。
            依次查找?guī)讉€(gè)頭文件,不難發(fā)現(xiàn),BSTR被typedef成OLECHAR *,而OLECHAR被typedef成WCHAR(未定義OLE2ANSI預(yù)處理宏)或char (定義OLE2ANSI預(yù)處理宏);繼續(xù)順藤摸瓜,WCHAR被typedef成wchar_t,而wchar_t被typedef成unsigned short。

            最終的結(jié)果就是,BSTR被typedef成unsigned short *(未定義OLE2ANSI預(yù)處理宏)或char *(定義OLE2ANSI預(yù)處理宏)。

            原來(lái)BSTR只是一個(gè)指針。

            本文不考慮定義預(yù)處理宏OLE2ANSI的情況。

            在COM編程中,凡是使用BSTR的地方,我們可以使用類CComBSTR來(lái)代替BSTR。CComBSTR封裝了BSTR,通過(guò)各種重載構(gòu)造函數(shù)和操作符重載,僅僅使用對(duì)象定義,type cast等簡(jiǎn)單語(yǔ)句,就可以實(shí)現(xiàn)BSTR與各種類型字符串,包括LPSTR,LPOLESTR之間的轉(zhuǎn)換,賦值,連接,比較等操作。

            二 ATL字符串轉(zhuǎn)換宏
            除了CComBSTR類支持各種字符串類型到BSTR的轉(zhuǎn)換以外,ATL還有一組字符串轉(zhuǎn)換宏,可以方便的進(jìn)行各種類型字符串之間的轉(zhuǎn)換。這些宏有統(tǒng)一的形式:X2[C]Y或Z2BSTR。其中X,Y,Z可以為A,W,T,OLE中的任一種,X與Y不相同;C表示轉(zhuǎn)換的目標(biāo)類型為const——因此,這樣的組合方式總共有4x3x2+4=28種。

            根據(jù)預(yù)處理宏_UNICODE定義與否,T被替代成W或A,所以這28個(gè)轉(zhuǎn)換宏最終可以歸結(jié)為兩類:一類是A和W之間的轉(zhuǎn)換,另一類是A,W到BSTR的轉(zhuǎn)換。

            下面看看第一類中A2W的實(shí)現(xiàn)。
            // Exerpt from ATLCONV.H   
            #define A2W(lpa) (\
                    ((_lpa 
            = lpa) == NULL) ? NULL : (\
                        _convert 
            = (lstrlenA(_lpa)+1),\
                        ATLA2WHELPER((LPWSTR) alloca(_convert
            *2), _lpa, _convert)))

            A2W是支持從ANSI字符串到UNICODE字符串的轉(zhuǎn)換宏。可以看到,A2W的主體是一個(gè)?:表達(dá)式,根據(jù)源字符串是否為NULL,對(duì)冒號(hào)之前或之后的表達(dá)式進(jìn)行求值。冒號(hào)后的部分又是一個(gè)逗號(hào)表達(dá)式:先計(jì)算源字符串的長(zhǎng)度,再調(diào)用被定義為同名宏(大小寫不同)的轉(zhuǎn)換函數(shù),轉(zhuǎn)換函數(shù)中調(diào)用MultiByteToWideChar系統(tǒng)函數(shù)進(jìn)行具體的轉(zhuǎn)換,逗號(hào)表達(dá)式的值是轉(zhuǎn)換函數(shù)的返回值。最終,轉(zhuǎn)換宏的值是NULL或者轉(zhuǎn)換函數(shù)的返回值。

            值得注意的是,在A2W宏定義中引用了類似_lpa, _convert等標(biāo)識(shí)符,那么這些標(biāo)識(shí)符是什么,又是哪里來(lái)的呢?
            這就是為什么在使用這些字符串轉(zhuǎn)換宏之前,需要統(tǒng)一加上一句
            USES_CONVERSION;
            USES_CONVERSION也是宏定義,就是用來(lái)聲明轉(zhuǎn)換宏中使用的標(biāo)識(shí)符的。

            另外,MSDN中特別提到,A和W之間的轉(zhuǎn)換(A2W,W2A),以及通過(guò)預(yù)處理宏_UNICODE翻譯成A和W之間轉(zhuǎn)換的宏,它們無(wú)一例外的都是從調(diào)用函數(shù)棧上分配空間存放轉(zhuǎn)換結(jié)果字符串。因此,轉(zhuǎn)換結(jié)果字符串在調(diào)用函數(shù)返回后自動(dòng)被清除,也就是不能保留轉(zhuǎn)換結(jié)果用于調(diào)用函數(shù)外使用。

            與之相對(duì),再看看第二類,A,W,T,OLE到BSTR的轉(zhuǎn)換宏實(shí)現(xiàn)。
            // Exerpt from ATLCONV.H
            inline BSTR OLE2BSTR(LPCOLESTR lp) {return ::SysAllocString(lp);}
            #if defined(_UNICODE)
            // in these cases the default (TCHAR) is the same as OLECHAR
                inline BSTR T2BSTR(LPCTSTR lp) {return ::SysAllocString(lp);}
                inline BSTR A2BSTR(LPCSTR lp) {USES_CONVERSION; 
            return A2WBSTR(lp);}
                inline BSTR W2BSTR(LPCWSTR lp) {
            return ::SysAllocString(lp);}
            #elif defined(OLE2ANSI)
            // in these cases the default (TCHAR) is the same as OLECHAR
                inline BSTR T2BSTR(LPCTSTR lp) {return ::SysAllocString(lp);}
                inline BSTR A2BSTR(LPCSTR lp) {
            return ::SysAllocString(lp);}
                inline BSTR W2BSTR(LPCWSTR lp) {USES_CONVERSION; 
            return ::SysAllocString(W2COLE(lp));}
            #else
                inline BSTR T2BSTR(LPCTSTR lp) {USES_CONVERSION; 
            return A2WBSTR(lp);}
                inline BSTR A2BSTR(LPCSTR lp) {USES_CONVERSION; 
            return A2WBSTR(lp);}
                inline BSTR W2BSTR(LPCWSTR lp) {
            return ::SysAllocString(lp);}
            #endif

            同樣,除了OLE2BSTR以外,A,W,T到BSTR的轉(zhuǎn)換根據(jù)是否定義預(yù)處理宏_UNICODE進(jìn)行不同的處理,最終歸結(jié)為兩種類型的轉(zhuǎn)換:A2BSTR和W2BSTR。對(duì)于A2BSTR,由于BSTR也就是W,所以A2BSTR的轉(zhuǎn)換在理論上同A2W應(yīng)該相同。

            但實(shí)際上,A2BSTR調(diào)用函數(shù)A2WBSTR,A2WBSTR內(nèi)部從堆上分配空間,再調(diào)用系統(tǒng)轉(zhuǎn)換函數(shù)MultiByteToWideChar進(jìn)行轉(zhuǎn)換——這就是A,W之間的轉(zhuǎn)換與2BSTR類型的轉(zhuǎn)換的根本不同之處;W2BSTR就簡(jiǎn)單了,由于BSTR即是W字符串,因此不需要實(shí)際轉(zhuǎn)換,只需分配空間并拷貝源串作為轉(zhuǎn)換結(jié)果即可。

            對(duì)于OLE2BSTR,由于BSTR是從OLECHAR定義來(lái)的(見(jiàn)上面BSTR類型定義),因此不管預(yù)處理宏如何定義(包括OLE2ANSI是否定義),二者的類型始終是一致的,因此,從OLE到BSTR,并不需要進(jìn)行實(shí)際的轉(zhuǎn)換,只需分配空間并拷貝源串作為轉(zhuǎn)換結(jié)果即可。

            總結(jié)下來(lái),28個(gè)字符串轉(zhuǎn)換宏中,根據(jù)結(jié)果字符串存放位置分為兩類,一類是A,W之間的轉(zhuǎn)換宏,這一類宏的轉(zhuǎn)換結(jié)果字符串放在調(diào)用函數(shù)的棧空間,調(diào)用函數(shù)返回會(huì)該空間自動(dòng)清除,第二類為目的類型為BSTR的轉(zhuǎn)換宏,其轉(zhuǎn)換結(jié)果字符串放在系統(tǒng)堆,在調(diào)用函數(shù)返回后結(jié)果字符串仍然存在。這是兩類轉(zhuǎn)換宏之間的最顯著差別。

            測(cè)試代碼
            // TestATLX2Y.cpp

            #include 
            <iostream>
            #include 
            <atlbase.h>
            using namespace std;

            #ifdef _UNICODE
            #define TCOUT wcout
            #else
            #define TCOUT cout
            #endif

            int main()
            {
            #ifdef _UNICODE
                TCOUT 
            << _T("----- Test with _UNICODE --------"<< endl;
            #else
                TCOUT 
            << _T("----- Test without _UNICODE --------"<< endl;
            #endif

                LPTSTR lptstr
            =_T("TCHAR is either char or wchar_t");
                LPSTR lpstr
            ="How can I indicate i'm a LPSTR?";
                LPWSTR lpwstr
            =L"I am a wide-character string";
                TCOUT 
            << _T("lptstr:"<< lptstr << endl;
                cout 
            << "lpstr:" << lpstr << endl;
                wcout 
            << L"lpwstr:" << lpwstr << endl;
                
                USES_CONVERSION;
                TCOUT 
            << _T("A2T:"<< A2T(lpstr) << endl;
                wcout 
            << L"A2W:" << A2W(lpstr) << endl;
                cout 
            << "T2A:" << T2A(lptstr) << endl;
                wcout 
            << L"T2W:" << T2W(lptstr) << endl;
                cout 
            << "W2A:" << W2A(lpwstr) << endl;
                TCOUT 
            << _T("W2T:"<< W2T(lpwstr) << endl;
                BSTR bstr;
                wcout 
            << L"A2BSTR:" << (bstr=A2BSTR(lpstr)) << endl;
                ::SysFreeString(bstr);
                wcout 
            << L"W2BSTR:" << (bstr=W2BSTR(lpwstr)) << endl;
                
                wcout 
            << L"T2BSTR:" << (bstr=T2BSTR(lptstr)) << endl;
                
            // How to know if we need to free the space pointered by returned value of T2BSTR?
                if((void *)bstr!=(void *)lptstr) ::SysFreeString(bstr);
                
                wcout 
            << L"OLE2BSTR:" << OLE2BSTR(lpwstr) << endl;
                
            return 0;
            }


            代碼運(yùn)行結(jié)果:

            posted on 2011-12-03 22:56 小蔥蘸醬 閱讀(5301) 評(píng)論(0)  編輯 收藏 引用 所屬分類: COM

            99热精品久久只有精品| 亚洲综合熟女久久久30p| 久久婷婷久久一区二区三区| 久久精品国产亚洲av影院| 久久亚洲欧美日本精品| 久久影院午夜理论片无码 | 日韩精品久久无码中文字幕| 国产精品一久久香蕉产线看| 一本一道久久a久久精品综合| 伊人久久大香线蕉av不卡| 久久国产高清字幕中文| 狠狠色丁香久久婷婷综合蜜芽五月 | 人妻丰满AV无码久久不卡| 国产午夜福利精品久久| 久久国产免费直播| 国产精品99久久精品爆乳| 嫩草伊人久久精品少妇AV| 久久93精品国产91久久综合| 亚洲午夜久久久久久久久久| 免费精品久久久久久中文字幕| 成人久久精品一区二区三区| 亚洲第一极品精品无码久久| 久久久久久久综合综合狠狠| 99re这里只有精品热久久| 香蕉久久永久视频| 久久天天日天天操综合伊人av| 91精品国产乱码久久久久久| 一本久久知道综合久久| 久久综合久久综合亚洲| 日韩AV毛片精品久久久| 久久国产热这里只有精品| 久久精品国产精品亚洲精品| 国产精品久久久久久影院| 久久国产热精品波多野结衣AV| 中文字幕无码免费久久| 久久亚洲AV无码精品色午夜麻豆 | 久久久久亚洲AV无码专区首JN| 久久人搡人人玩人妻精品首页| 久久精品国产72国产精福利| 久久久久国产一区二区| 久久久久无码国产精品不卡|