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

<2009年4月>
2930311234
567891011
12131415161718
19202122232425
262728293012
3456789

統計

  • 隨筆 - 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 小蔥蘸醬 閱讀(5315) 評論(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在线播放一区二区三区| 久久人人爽国产| 欧美国产日韩一区二区在线观看 | 亚洲高清一区二| 另类天堂视频在线观看| 欧美激情精品久久久久久蜜臀| 最近中文字幕日韩精品| 一区二区三区国产在线| 午夜综合激情| 欧美激情第六页| 国产精品视频免费观看| 一区二区自拍| 亚洲视频自拍偷拍| 久久日韩精品| 99视频精品在线| 久久久久高清| 国产精品高清免费在线观看| 在线精品视频一区二区| 亚洲免费网址| 亚洲第一区在线| 午夜精品一区二区三区在线播放 | 一区二区三区四区五区在线 | 国产精品成人一区二区三区吃奶 | 久久国产精品黑丝| 亚洲欧美日韩视频一区| 国产一区二区日韩精品欧美精品| 亚洲国产精品www| 亚洲一区二区三区在线看 | 91久久久在线| 欧美一区二区高清在线观看| 免费在线观看日韩欧美| 国产欧美日韩综合精品二区| 99re6这里只有精品视频在线观看| 午夜在线精品偷拍| 亚洲精品网址在线观看| 久久综合网络一区二区| 国产日韩在线看| 亚洲欧美日韩精品一区二区| 亚洲日本免费| 欧美aⅴ一区二区三区视频| 黄色一区二区在线观看| 欧美在线观看视频| 亚洲一区二区三区午夜| 欧美视频一区二区三区四区| 99精品久久久| 亚洲精品美女在线观看| 欧美激情久久久久久| 在线观看视频欧美| 老鸭窝毛片一区二区三区| 久久se精品一区精品二区| 国产模特精品视频久久久久| 亚洲欧美日韩国产一区二区| 正在播放亚洲| 国产精品麻豆成人av电影艾秋| 亚洲天堂av高清| 亚洲天堂网站在线观看视频| 欧美亚男人的天堂| 亚洲综合第一页| 亚洲在线第一页| 国产欧美一区二区白浆黑人| 久久成人18免费网站| 欧美一区二区三区精品| 国内成人精品视频| 另类综合日韩欧美亚洲| 免费在线观看日韩欧美| 洋洋av久久久久久久一区| 亚洲精品久久久蜜桃| 欧美午夜在线观看| 久久国产66| 久久成人国产| 亚洲精品欧美日韩| 一级成人国产| 国产在线精品一区二区夜色| 美腿丝袜亚洲色图| 欧美激情第五页| 亚洲男人第一av网站| 欧美一区二区三区久久精品| 亚洲第一综合天堂另类专| 亚洲欧洲一区| 国产精品免费福利| 裸体丰满少妇做受久久99精品| 欧美成人免费网| 午夜精品久久久久久久99水蜜桃| 亚洲激情精品| 久久久av水蜜桃| 久久久精品一区| 99re热精品| 亚洲欧美日韩综合一区| 亚洲第一偷拍| 一区二区不卡在线视频 午夜欧美不卡在| 国产精品久久久爽爽爽麻豆色哟哟| 欧美在线免费观看视频| 免费观看成人| 欧美专区18| 欧美精品日韩| 老**午夜毛片一区二区三区| 欧美日韩一区二区三区在线看| 久久久久久久999| 欧美日韩国产色综合一二三四| 久久精品国产亚洲一区二区三区| 欧美大片国产精品| 久久精品视频导航| 欧美日韩国产一级| 欧美插天视频在线播放| 国产片一区二区| 一区二区久久久久| 亚洲精品日韩激情在线电影| 欧美一区二区在线免费观看| 一本大道久久a久久精品综合 | 亚洲欧美日韩国产综合| 亚洲精品在线观| 久久久蜜桃一区二区人| 欧美在线免费视频| 欧美无砖砖区免费| 亚洲精选久久| 亚洲精品在线视频观看| 久久婷婷国产麻豆91天堂| 欧美一区二区在线免费观看 | 欧美.日韩.国产.一区.二区| 久久久噜噜噜久久中文字免| 国产精品理论片在线观看| 亚洲精品在线观看视频| 亚洲人成网站999久久久综合| 久久久蜜桃一区二区人| 久久久久久久久久久久久女国产乱 | 一区二区三区高清在线| 免费在线观看一区二区| 欧美高清视频www夜色资源网| 狠狠色丁香婷综合久久| 欧美制服丝袜| 久久伊人精品天天| 亚洲成色777777在线观看影院| 欧美一区二区三区四区在线| 久久精品国产精品亚洲综合| 国产亚洲免费的视频看| 欧美伊久线香蕉线新在线| 久久精品一区二区三区中文字幕| 国产日韩一区二区三区在线| 欧美在线观看视频一区二区三区 | 欧美福利视频网站| 在线日韩欧美视频| 玖玖视频精品| 91久久黄色| 久久精品色图| 亚洲国产电影| 免播放器亚洲| 亚洲免费av片| 欧美一级成年大片在线观看| 国产精品影片在线观看| 久久激情视频久久| 亚洲国产另类久久精品| 亚洲午夜一区二区三区| 国产免费亚洲高清| 玖玖玖免费嫩草在线影院一区| 亚洲人成网站777色婷婷| 亚洲综合色丁香婷婷六月图片| 国产精品一二三视频| 久久精品最新地址| 亚洲人在线视频| 亚洲欧美日韩专区| 国产亚洲欧美在线| 欧美 亚欧 日韩视频在线| 亚洲天堂av图片| 免费观看国产成人| 亚洲一区视频在线观看视频| 黄色精品一区二区| 欧美午夜电影网| 久久在线免费观看| 亚洲午夜伦理| 欧美国产日韩xxxxx| 午夜精品视频在线| 亚洲国产精品毛片| 国产欧美欧美| 欧美国产日本在线| 午夜精品在线视频| 亚洲精品一区二区三区在线观看 | 亚洲图中文字幕| 欧美99久久| 久久成人18免费网站| 9色porny自拍视频一区二区| 狠狠色综合色区| 国产精品成人免费| 欧美日本韩国一区| 久久伊人一区二区| 午夜精品久久久久久久蜜桃app | 久久国产天堂福利天堂| 一本久道久久综合中文字幕| 老司机一区二区三区| 亚洲欧洲99久久| 一区二区日韩伦理片| 亚洲经典自拍| 在线成人激情黄色| 国产亚洲一本大道中文在线| 欧美日韩激情小视频| 午夜久久久久久|