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

            C++ Programmer's Cookbook

            {C++ 基礎(chǔ)} {C++ 高級(jí)} {C#界面,C++核心算法} {設(shè)計(jì)模式} {C#基礎(chǔ)}

            Visual C++.NET中 字符串轉(zhuǎn)換方法

            Visual C++.NET中 字符串轉(zhuǎn)換方法
            2002-12-06 14:48:39

            丁有和

              Visual C++.NET涉及到ATL/ATL Server、MFC和托管C++等多種編程方式,不僅功能強(qiáng)大而且應(yīng)用廣泛。在編程中,我們常常會(huì)遇到ANSI、Unicode以及BSTR不同編碼類型的字符串轉(zhuǎn)換操作。本文先介紹基本字符串類型,然后說明相關(guān)的類,如CComBSTR、_bstr_t、CStringT等,最后討論它們的轉(zhuǎn)換方法,其中還包括使用最新ATL7.0的轉(zhuǎn)換類和宏,如CA2CT、CA2TEX等。

              一、BSTR、LPSTR和LPWSTR

              在Visual C++.NET的所有編程方式中,我們常常要用到這樣的一些基本字符串類型,如BSTR、LPSTR和LPWSTR等。之所以出現(xiàn)類似上述的這些數(shù)據(jù)類型,是因?yàn)椴煌幊陶Z言之間的數(shù)據(jù)交換以及對(duì)ANSI、Unicode和多字節(jié)字符集(MBCS)的支持。

              那么什么是BSTR、LPSTR以及LPWSTR呢?

              BSTR(Basic STRing,Basic字符串)是一個(gè)OLECHAR*類型的Unicode字符串。它被描述成一個(gè)與自動(dòng)化相兼容的類型。由于操作系統(tǒng)提供相應(yīng)的API函數(shù)(如SysAllocString)來管理它以及一些默認(rèn)的調(diào)度代碼,因此BSTR實(shí)際上就是一個(gè)COM字符串,但它卻在自動(dòng)化技術(shù)以外的多種場(chǎng)合下得到廣泛使用。圖1描述了BSTR的結(jié)構(gòu),其中DWORD值是字符串中實(shí)際所占用的字節(jié)數(shù),且它的值是字符串中Unicode字符的兩倍。

              LPSTR和LPWSTR是Win32和VC++所使用的一種字符串?dāng)?shù)據(jù)類型。LPSTR被定義成是一個(gè)指向以NULL(‘\0’)結(jié)尾的8位ANSI字符數(shù)組指針,而LPWSTR是一個(gè)指向以NULL結(jié)尾的16位雙字節(jié)字符數(shù)組指針。在VC++中,還有類似的字符串類型,如LPTSTR、LPCTSTR等,它們的含義如圖2所示。

              例如,LPCTSTR是指“l(fā)ong pointer to a constant generic string”,表示“一個(gè)指向一般字符串常量的長指針類型”,與C/C++的const char*相映射,而LPTSTR映射為 char*。

              一般地,還有下列類型定義:

            #ifdef UNICODE
             typedef LPWSTR LPTSTR;
             typedef LPCWSTR LPCTSTR;
            #else
             typedef LPSTR LPTSTR;
             typedef LPCSTR LPCTSTR;
            #endif 

              二、CString、CStringA 和 CStringW

              Visual C++.NET中將CStringT作為ATL和MFC的共享的“一般”字符串類,它有CString、CStringA和CStringW三種形式,分別操作不同字符類型的字符串。這些字符類型是TCHAR、char和wchar_t。TCHAR在Unicode平臺(tái)中等同于WCHAR(16位Unicode字符),在ANSI中等價(jià)于char。wchar_t通常定義為unsigned short。由于CString在MFC應(yīng)用程序中經(jīng)常用到,這里不再重復(fù)。

              三、VARIANT、COleVariant 和_variant_t

              在OLE、ActiveX和COM中,VARIANT數(shù)據(jù)類型提供了一種非常有效的機(jī)制,由于它既包含了數(shù)據(jù)本身,也包含了數(shù)據(jù)的類型,因而它可以實(shí)現(xiàn)各種不同的自動(dòng)化數(shù)據(jù)的傳輸。下面讓我們來看看OAIDL.H文件中VARIANT定義的一個(gè)簡化版:

            struct tagVARIANT {
             VARTYPE vt;
             union {
              short iVal; // VT_I2.
              long lVal; // VT_I4.
              float fltVal; // VT_R4.
              double dblVal; // VT_R8.
              DATE date; // VT_DATE.
              BSTR bstrVal; // VT_BSTR.
              …
              short * piVal; // VT_BYREF|VT_I2.
              long * plVal; // VT_BYREF|VT_I4.
              float * pfltVal; // VT_BYREF|VT_R4.
              double * pdblVal; // VT_BYREF|VT_R8.
              DATE * pdate; // VT_BYREF|VT_DATE.
              BSTR * pbstrVal; // VT_BYREF|VT_BSTR.
             };
            };

              顯然,VARIANT類型是一個(gè)C結(jié)構(gòu),它包含了一個(gè)類型成員vt、一些保留字節(jié)以及一個(gè)大的union類型。例如,如果vt為VT_I2,那么我們可以從iVal中讀出VARIANT的值。同樣,當(dāng)給一個(gè)VARIANT變量賦值時(shí),也要先指明其類型。例如:

            VARIANT va;
            :: VariantInit(&va); // 初始化
            int a = 2002;
            va.vt = VT_I4; // 指明long數(shù)據(jù)類型
            va.lVal = a; // 賦值

              為了方便處理VARIANT類型的變量,Windows還提供了這樣一些非常有用的函數(shù):

              VariantInit —— 將變量初始化為VT_EMPTY;

              VariantClear —— 消除并初始化VARIANT;

              VariantChangeType —— 改變VARIANT的類型;

              VariantCopy —— 釋放與目標(biāo)VARIANT相連的內(nèi)存并復(fù)制源VARIANT。

              COleVariant類是對(duì)VARIANT結(jié)構(gòu)的封裝。它的構(gòu)造函數(shù)具有極為強(qiáng)大大的功能,當(dāng)對(duì)象構(gòu)造時(shí)首先調(diào)用VariantInit進(jìn)行初始化,然后根據(jù)參數(shù)中的標(biāo)準(zhǔn)類型調(diào)用相應(yīng)的構(gòu)造函數(shù),并使用VariantCopy進(jìn)行轉(zhuǎn)換賦值操作,當(dāng)VARIANT對(duì)象不在有效范圍時(shí),它的析構(gòu)函數(shù)就會(huì)被自動(dòng)調(diào)用,由于析構(gòu)函數(shù)調(diào)用了VariantClear,因而相應(yīng)的內(nèi)存就會(huì)被自動(dòng)清除。除此之外,COleVariant的賦值操作符在與VARIANT類型轉(zhuǎn)換中為我們提供極大的方便。例如下面的代碼:

            COleVariant v1("This is a test"); // 直接構(gòu)造
            COleVariant v2 = "This is a test";
            // 結(jié)果是VT_BSTR類型,值為"This is a test"
            COleVariant v3((long)2002);
            COleVariant v4 = (long)2002;
            // 結(jié)果是VT_I4類型,值為2002

              _variant_t是一個(gè)用于COM的VARIANT類,它的功能與COleVariant相似。不過在Visual C++.NET的MFC應(yīng)用程序中使用時(shí)需要在代碼文件前面添加下列兩句:

              #include "comutil.h"

              #pragma comment( lib, "comsupp.lib" )

              四、CComBSTR和_bstr_t

              CComBSTR是對(duì)BSTR數(shù)據(jù)類型封裝的一個(gè)ATL類,它的操作比較方便。例如:

            CComBSTR bstr1;
            bstr1 = "Bye"; // 直接賦值
            OLECHAR* str = OLESTR("ta ta"); // 長度為5的寬字符
            CComBSTR bstr2(wcslen(str)); // 定義長度為5
            wcscpy(bstr2.m_str, str); // 將寬字符串復(fù)制到BSTR中
            CComBSTR bstr3(5, OLESTR("Hello World"));
            CComBSTR bstr4(5, "Hello World");
            CComBSTR bstr5(OLESTR("Hey there"));
            CComBSTR bstr6("Hey there");
            CComBSTR bstr7(bstr6);
            // 構(gòu)造時(shí)復(fù)制,內(nèi)容為"Hey there"

              _bstr_t是是C++對(duì)BSTR的封裝,它的構(gòu)造和析構(gòu)函數(shù)分別調(diào)用SysAllocString和SysFreeString函數(shù),其他操作是借用BSTR API函數(shù)。與_variant_t相似,使用時(shí)也要添加comutil.h和comsupp.lib。


             
             

              五、BSTR、char*和CString轉(zhuǎn)換

              (1) char*轉(zhuǎn)換成CString

              若將char*轉(zhuǎn)換成CString,除了直接賦值外,還可使用CString::Format進(jìn)行。例如:

            char chArray[] = "This is a test";
            char * p = "This is a test";

              或

            LPSTR p = "This is a test";

              或在已定義Unicode應(yīng)的用程序中

            TCHAR * p = _T("This is a test");

              或

            LPTSTR p = _T("This is a test");
            CString theString = chArray;
            theString.Format(_T("%s"), chArray);
            theString = p;

              (2) CString轉(zhuǎn)換成char*

              若將CString類轉(zhuǎn)換成char*(LPSTR)類型,常常使用下列三種方法:

              方法一,使用強(qiáng)制轉(zhuǎn)換。例如:

            CString theString( "This is a test" );
            LPTSTR lpsz =(LPTSTR)(LPCTSTR)theString; 

              方法二,使用strcpy。例如:

            CString theString( "This is a test" );
            LPTSTR lpsz = new TCHAR[theString.GetLength()+1];
            _tcscpy(lpsz, theString);

              需要說明的是,strcpy(或可移值Unicode/MBCS的_tcscpy)的第二個(gè)參數(shù)是 const wchar_t* (Unicode)或const char* (ANSI),系統(tǒng)編譯器將會(huì)自動(dòng)對(duì)其進(jìn)行轉(zhuǎn)換。

              方法三,使用CString::GetBuffer。例如:

            CString s(_T("This is a test "));
            LPTSTR p = s.GetBuffer();
            // 在這里添加使用p的代碼
            if(p != NULL) *p = _T('\0');
            s.ReleaseBuffer();
            // 使用完后及時(shí)釋放,以便能使用其它的CString成員函數(shù)

              (3) BSTR轉(zhuǎn)換成char*

              方法一,使用ConvertBSTRToString。例如:

            #include
            #pragma comment(lib, "comsupp.lib")
            int _tmain(int argc, _TCHAR* argv[]){
            BSTR bstrText = ::SysAllocString(L"Test");
            char* lpszText2 = _com_util::ConvertBSTRToString(bstrText);
            SysFreeString(bstrText); // 用完釋放
            delete[] lpszText2;
            return 0;

              方法二,使用_bstr_t的賦值運(yùn)算符重載。例如:

            _bstr_t b = bstrText;
            char* lpszText2 = b;

              (4) char*轉(zhuǎn)換成BSTR

              方法一,使用SysAllocString等API函數(shù)。例如:

            BSTR bstrText = ::SysAllocString(L"Test");
            BSTR bstrText = ::SysAllocStringLen(L"Test",4);
            BSTR bstrText = ::SysAllocStringByteLen("Test",4);

              方法二,使用COleVariant或_variant_t。例如:

            //COleVariant strVar("This is a test");
            _variant_t strVar("This is a test");
            BSTR bstrText = strVar.bstrVal;

              方法三,使用_bstr_t,這是一種最簡單的方法。例如:

            BSTR bstrText = _bstr_t("This is a test");

              方法四,使用CComBSTR。例如:

            BSTR bstrText = CComBSTR("This is a test");

              或

            CComBSTR bstr("This is a test");
            BSTR bstrText = bstr.m_str;

              方法五,使用ConvertStringToBSTR。例如:

            char* lpszText = "Test";
            BSTR bstrText = _com_util::ConvertStringToBSTR(lpszText);

              (5) CString轉(zhuǎn)換成BSTR

              通常是通過使用CStringT::AllocSysString來實(shí)現(xiàn)。例如:

            CString str("This is a test");
            BSTR bstrText = str.AllocSysString();

            SysFreeString(bstrText); // 用完釋放 

              (6) BSTR轉(zhuǎn)換成CString

              一般可按下列方法進(jìn)行:

            BSTR bstrText = ::SysAllocString(L"Test");
            CStringA str;
            str.Empty();
            str = bstrText; 

              或

            CStringA str(bstrText);

              (7) ANSI、Unicode和寬字符之間的轉(zhuǎn)換

              方法一,使用MultiByteToWideChar將ANSI字符轉(zhuǎn)換成Unicode字符,使用WideCharToMultiByte將Unicode字符轉(zhuǎn)換成ANSI字符。

              方法二,使用“_T”將ANSI轉(zhuǎn)換成“一般”類型字符串,使用“L”將ANSI轉(zhuǎn)換成Unicode,而在托管C++環(huán)境中還可使用S將ANSI字符串轉(zhuǎn)換成String*對(duì)象。例如:

            TCHAR tstr[] = _T("this is a test");
            wchar_t wszStr[] = L"This is a test";
            String* str = S”This is a test”;

              方法三,使用ATL 7.0的轉(zhuǎn)換宏和類。ATL7.0在原有3.0基礎(chǔ)上完善和增加了許多字符串轉(zhuǎn)換宏以及提供相應(yīng)的類
            ,它具有如圖3所示的統(tǒng)一形式:

              其中,第一個(gè)C表示“類”,以便于ATL 3.0宏相區(qū)別,第二個(gè)C表示常量,2表示“to”,EX表示要開辟一定大
            小的緩沖。SourceType和DestinationType可以是A、T、W和OLE,其含義分別是ANSI、Unicode、“一般”類型和
            OLE字符串。例如,CA2CT就是將ANSI轉(zhuǎn)換成一般類型的字符串常量。下面是一些示例代碼:

            LPTSTR tstr= CA2TEX<16>("this is a test");
            LPCTSTR tcstr= CA2CT("this is a test");
            wchar_t wszStr[] = L"This is a test";
            char* chstr = CW2A(wszStr); 

              六、結(jié)語

              幾乎所有的程序都要用到字符串,而Visual C++.NET由于功能強(qiáng)大、應(yīng)用廣泛,因而字符串之間的轉(zhuǎn)換更為頻繁
            。本文幾乎涉及到目前的所有轉(zhuǎn)換方法。當(dāng)然對(duì)于.NET框架來說,還可使用Convert和Text類進(jìn)行不同數(shù)據(jù)類型以及
            字符編碼之間的相互轉(zhuǎn)換。

            posted on 2005-12-23 12:52 夢(mèng)在天涯 閱讀(1452) 評(píng)論(0)  編輯 收藏 引用 所屬分類: CPlusPlusMFC/QTCOM/ATL

            公告

            EMail:itech001#126.com

            導(dǎo)航

            統(tǒng)計(jì)

            • 隨筆 - 461
            • 文章 - 4
            • 評(píng)論 - 746
            • 引用 - 0

            常用鏈接

            隨筆分類

            隨筆檔案

            收藏夾

            Blogs

            c#(csharp)

            C++(cpp)

            Enlish

            Forums(bbs)

            My self

            Often go

            Useful Webs

            Xml/Uml/html

            搜索

            •  

            積分與排名

            • 積分 - 1804173
            • 排名 - 5

            最新評(píng)論

            閱讀排行榜

            91精品国产91久久久久福利| 久久国产劲爆AV内射—百度| 久久精品国产91久久综合麻豆自制| 久久精品免费观看| 99久久无码一区人妻| 久久亚洲AV成人出白浆无码国产| 久久国产V一级毛多内射| 97久久精品人妻人人搡人人玩| 99久久99久久久精品齐齐| 天天影视色香欲综合久久| 色综合久久综精品| 99久久99久久精品免费看蜜桃| 久久久久久国产a免费观看不卡 | 精品久久久久久国产三级| 亚洲欧美国产精品专区久久| 久久精品99无色码中文字幕| 伊人久久大香线蕉AV色婷婷色| 热久久最新网站获取| 亚洲人成网站999久久久综合| 狼狼综合久久久久综合网| 久久国产美女免费观看精品 | 欧美精品国产综合久久| 亚洲精品乱码久久久久久不卡| 99久久精品午夜一区二区| a级成人毛片久久| 国产A级毛片久久久精品毛片| 青青热久久国产久精品| 97超级碰碰碰碰久久久久| 久久99国产精一区二区三区| 久久亚洲美女精品国产精品| 欧美粉嫩小泬久久久久久久 | 亚洲欧洲精品成人久久曰影片 | 国产69精品久久久久99尤物| 国产69精品久久久久99| 狠狠色丁香久久婷婷综| 国产一久久香蕉国产线看观看| 久久久久久久久无码精品亚洲日韩 | 久久99精品国产99久久6| 狠狠狠色丁香婷婷综合久久五月| 国产亚洲色婷婷久久99精品| 久久精品人人做人人妻人人玩|