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

<2025年10月>
2829301234
567891011
12131415161718
19202122232425
2627282930311
2345678

統計

  • 隨筆 - 24
  • 文章 - 0
  • 評論 - 17
  • 引用 - 0

常用鏈接

留言簿(4)

隨筆分類

隨筆檔案

相冊

搜索

  •  

最新評論

閱讀排行榜

評論排行榜

BSTR CComBSTR及ATL字符串轉換宏
一 BSTR及CComBSTR

MSDN文檔中說,BSTR是四字節長度前綴的,NULL結尾的寬字符串,稱之為VB字符串或BINARY字符串。
依次查找幾個頭文件,不難發現,BSTR被typedef成OLECHAR *,而OLECHAR被typedef成WCHAR(未定義OLE2ANSI預處理宏)或char (定義OLE2ANSI預處理宏);繼續順藤摸瓜,WCHAR被typedef成wchar_t,而wchar_t被typedef成unsigned short。

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

原來BSTR只是一個指針。

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

在COM編程中,凡是使用BSTR的地方,我們可以使用類CComBSTR來代替BSTR。CComBSTR封裝了BSTR,通過各種重載構造函數和操作符重載,僅僅使用對象定義,type cast等簡單語句,就可以實現BSTR與各種類型字符串,包括LPSTR,LPOLESTR之間的轉換,賦值,連接,比較等操作。

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

根據預處理宏_UNICODE定義與否,T被替代成W或A,所以這28個轉換宏最終可以歸結為兩類:一類是A和W之間的轉換,另一類是A,W到BSTR的轉換。

下面看看第一類中A2W的實現。
// 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字符串的轉換宏。可以看到,A2W的主體是一個?:表達式,根據源字符串是否為NULL,對冒號之前或之后的表達式進行求值。冒號后的部分又是一個逗號表達式:先計算源字符串的長度,再調用被定義為同名宏(大小寫不同)的轉換函數,轉換函數中調用MultiByteToWideChar系統函數進行具體的轉換,逗號表達式的值是轉換函數的返回值。最終,轉換宏的值是NULL或者轉換函數的返回值。

值得注意的是,在A2W宏定義中引用了類似_lpa, _convert等標識符,那么這些標識符是什么,又是哪里來的呢?
這就是為什么在使用這些字符串轉換宏之前,需要統一加上一句
USES_CONVERSION;
USES_CONVERSION也是宏定義,就是用來聲明轉換宏中使用的標識符的。

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

與之相對,再看看第二類,A,W,T,OLE到BSTR的轉換宏實現。
// 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的轉換根據是否定義預處理宏_UNICODE進行不同的處理,最終歸結為兩種類型的轉換:A2BSTR和W2BSTR。對于A2BSTR,由于BSTR也就是W,所以A2BSTR的轉換在理論上同A2W應該相同。

但實際上,A2BSTR調用函數A2WBSTR,A2WBSTR內部從堆上分配空間,再調用系統轉換函數MultiByteToWideChar進行轉換——這就是A,W之間的轉換與2BSTR類型的轉換的根本不同之處;W2BSTR就簡單了,由于BSTR即是W字符串,因此不需要實際轉換,只需分配空間并拷貝源串作為轉換結果即可。

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

總結下來,28個字符串轉換宏中,根據結果字符串存放位置分為兩類,一類是A,W之間的轉換宏,這一類宏的轉換結果字符串放在調用函數的棧空間,調用函數返回會該空間自動清除,第二類為目的類型為BSTR的轉換宏,其轉換結果字符串放在系統堆,在調用函數返回后結果字符串仍然存在。這是兩類轉換宏之間的最顯著差別。

測試代碼
// 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;
}


代碼運行結果:

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

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            老色批av在线精品| 亚洲欧美日韩一区| 欧美激情第二页| 日韩午夜精品| 欧美亚洲成人免费| 午夜日韩福利| 亚洲美女视频在线免费观看| 亚洲欧美日韩天堂| 午夜视黄欧洲亚洲| 激情懂色av一区av二区av| 欧美电影免费观看高清完整版| 久久大综合网| 亚洲国产日韩一级| 国产精品尤物| 久久久久久久一区二区三区| 一区二区三区|亚洲午夜| 欧美日韩一卡二卡| 久久精品欧洲| 99精品欧美| 久久人人97超碰精品888| 亚洲一级在线观看| 亚洲国产欧美在线人成| 亚洲精品在线二区| 亚洲第一精品在线| 国外成人免费视频| 国产亚洲一区二区在线观看| 欧美视频日韩视频| 欧美日韩视频第一区| 国产精品揄拍500视频| 一色屋精品视频免费看| 亚洲看片一区| 久久狠狠一本精品综合网| 亚洲欧美国产日韩中文字幕| 欧美一区二区三区的| 亚洲一区日韩| 一区二区三欧美| 久久久精品国产一区二区三区| 亚洲嫩草精品久久| 欧美国产亚洲另类动漫| 亚洲男人影院| 欧美在线观看www| 久久激情视频久久| 久久久久.com| 国产精品你懂的在线欣赏| 国产精品黄视频| 国产裸体写真av一区二区| 亚洲精品在线一区二区| 久久精品人人爽| 亚洲午夜免费视频| 久久福利资源站| 国产精品一级在线| 一本色道久久综合亚洲精品婷婷| 亚洲一区www| 亚洲精品小视频| 你懂的视频欧美| 欧美人与性禽动交情品| 国产精品超碰97尤物18| 亚洲人成绝费网站色www| 一区二区三区日韩在线观看| 免费久久久一本精品久久区| 日韩天堂在线视频| 欧美sm视频| 亚洲国产精品成人精品| 美女91精品| 亚洲精品欧美一区二区三区| 日韩午夜激情| 欧美高清在线视频| 欧美视频导航| 日韩一级大片| 亚洲人成亚洲人成在线观看图片| 国产精品婷婷午夜在线观看| 国产精品日韩久久久久| 99精品国产一区二区青青牛奶| 欧美黄在线观看| 免费日韩视频| 亚洲伦理精品| 99亚洲一区二区| 欧美午夜精品久久久久久人妖| 中文久久精品| 牛牛国产精品| 亚洲图片欧美日产| 国产精品一区二区久久| 久久精品综合一区| 六月婷婷一区| 亚洲午夜一区二区三区| 亚洲午夜电影| 激情视频一区二区| 亚洲国产成人av| 欧美在线观看日本一区| 狠狠色2019综合网| 亚洲人成网站777色婷婷| 欧美日韩在线视频一区二区| 午夜视频一区在线观看| 久久久无码精品亚洲日韩按摩| 亚洲人成7777| 亚洲欧美伊人| 国产精品国产三级国产专播精品人| 亚洲欧美在线高清| 久久人人97超碰精品888| 久久亚洲视频| 久久丁香综合五月国产三级网站| 激情五月婷婷综合| 欧美成人第一页| 欧美欧美午夜aⅴ在线观看| 午夜精品一区二区三区在线| 久久久久久网站| 亚洲免费av观看| 亚洲女与黑人做爰| 最新高清无码专区| 亚洲午夜久久久| 激情婷婷欧美| 亚洲永久视频| 欧美精品色综合| 日韩视频二区| 欧美亚洲三级| 国产午夜精品久久| 久久国产日本精品| 欧美日韩国产色视频| 国产综合自拍| 亚洲一区二区三区视频| 性欧美激情精品| 中文av一区二区| 久久亚洲国产精品一区二区| 亚洲欧美福利一区二区| 欧美成人一品| 欧美成人综合网站| 久久久精品国产免费观看同学| 美女诱惑一区| 久久久国产成人精品| 国产精品激情| 亚洲精品久久在线| 伊人久久综合97精品| 先锋亚洲精品| 欧美一区精品| 国产欧美不卡| 久久福利影视| 国产伦精品免费视频| 久久国产精品久久久久久久久久| 欧美黄在线观看| 欧美成熟视频| 亚洲高清一区二| 麻豆精品在线播放| 免费看的黄色欧美网站| 禁久久精品乱码| 久久久久综合网| 免费精品99久久国产综合精品| 黄色亚洲免费| 久久精品最新地址| 欧美成人免费网站| 亚洲人成人99网站| 欧美精品在线播放| 99精品欧美一区二区三区综合在线 | 亚洲欧美在线一区| 在线不卡亚洲| 午夜久久福利| 欧美日韩国产一中文字不卡| 91久久国产综合久久蜜月精品 | 久久综合电影一区| 欧美日韩国产丝袜另类| 91久久极品少妇xxxxⅹ软件| 亚洲免费观看在线观看| 欧美精品在线一区二区| 一区二区日韩| 在线欧美三区| 欧美精品xxxxbbbb| 亚洲午夜视频在线观看| 久久精品夜色噜噜亚洲aⅴ| 激情五月综合色婷婷一区二区| 香蕉国产精品偷在线观看不卡| 免费短视频成人日韩| 亚洲图片在区色| 美日韩精品视频| 亚洲精品免费一区二区三区| 亚洲视频第一页| 国产曰批免费观看久久久| 久久综合伊人| 一区二区精品在线| 久久综合伊人77777尤物| 欧美日韩一区二区在线视频 | 欧美电影在线| 亚洲性夜色噜噜噜7777| 国产一区二区三区电影在线观看| 美女精品国产| 亚洲欧美日韩精品久久奇米色影视| 免费久久久一本精品久久区| 亚洲女人av| 亚洲精品一区在线| 国产亚洲精品久久久久久| 欧美激情在线| 久久视频一区| 欧美日韩国产一中文字不卡| 国产免费亚洲高清| 你懂的国产精品永久在线| 一区二区三区三区在线| 亚洲二区视频| 国产欧美日韩91| 欧美日韩不卡一区| 免费成人激情视频| 欧美一区二区在线免费播放| 亚洲精品美女在线观看|