1. 程式人生 > >關於Char* ,CString ,WCHAR*之間的轉換問題

關於Char* ,CString ,WCHAR*之間的轉換問題

GDI+所有類的介面函式如果要傳遞字串作為引數的話,似乎都用UNICODE串,即WCHAR*。我開始也被整得暈頭轉向,因為視窗程式設計所用往往是CString,用IO流讀檔案資料又得到char *。得益於網上牛人們的總結,我用到以下幾種基本方法去實現三者間的轉換:


程式碼段一:
char * 轉WCHAR *:
::MultiByteToWideChar(CP_ACP,0,(const char *)res,int count,char * dest,int count);
類似地,WCHAR *轉char *:
WideCharToMultiByte(CP_ACP,   0,.........);
CString 轉WCHAR *:
wchar_t * p=str.AllocSysStrinig()
也有A2W(str)的,但是要包括ATL轉換標頭檔案#include;
並且在A2W前使用USES_CONVERSION巨集。
其它:
char*轉CString:
除了直接賦值外,還可使用CString::Format進行。
如char * p="sfdasf";
CString str=p; 或者str.Format("%s",p);
CString 轉char *
1.直接強制型別轉換:
 CString ss="sfasf";
  char *p=(LPSTR)(LPCSTR)ss;
2.CString::GetBuffer或LockBuffer
char * p=str.GetBuffer();
char * pt=str.LockBuffer();
WCHAR *轉CString
在網上沒有找到相關的文件,想想應該是可以直接賦值的
但是試驗發現雖無編譯錯誤,但是用中文的時候卻生亂碼,用字母的時候卻是正常,想其中肯定沒有錯誤,只是用MessageBox顯示的時候亂碼應該有其它原因,比如說使用了雙位元組字符集DBCS來顯示漢字(純屬猜測而已)。總體來說在Windows程式設計中:#define UNICODE
則CString .TCHAR,等均用的是UNICODE碼,一個字元佔兩個位元組。


 


程式碼段二:


 


CString m_strCarNo = "吉A13546";//實際中在我的程式裡"吉A13546"是從資料庫中讀的


WCHAR buffer[1024];
wcsset(buffer,0); 
MultiByteToWideChar(CP_ACP,0,(char*)LPCSTR(m_strCarNo),(int)strlen((char*)LPCSTR(m_strCarNo)),buffer,1024);
int nLength = wcslen(buffer);


SolidBrush brush(Color(255, 0, 0, 255));//顏色 字型
Gdiplus::Font font(L"Arial", 15, FontStyleRegular,UnitPixel);  
graphics.DrawString(buffer,nLength,&font,PointF(X-36,Y+16),&brush);


//注:font的定義也應如此




程式碼段三:


CString 是 MFC 的東西,轉換成WCHAR,要用 A2W,或 T2CW,要包含標頭檔案#include <atlconv.h>


CString str("Test");WCHAR buf[1024];


// 或自己動態分配USES_CONVERSION;


// 必須有這句wcscpy(buf, A2W((LPCSTR)str));


// 轉換,也可用 wcsncpy如果用SDK:


Font myFont(L"Arial", 16);


PointF origin(0.0f, 0.0f);


SolidBrush blackBrush(Color(255, 0, 0, 0));


// DrawString 有幾種引數設法,例如graphics.DrawString(buf,wcslen(buf),&myFont,origin,&blackBrush);


 


 


程式碼段四:


 


把char*轉換為wchar_t*
用stdlib.h中的mbstowcs_s函式,可以通過下面的例子瞭解其用法:
 
char *CStr = "string to convert";
size_t len = strlen(CStr) + 1;
size_t converted = 0;
wchar_t *WStr;
WStr=(wchar_t*)malloc(len*sizeof(wchar_t));
mbstowcs_s(&converted, WStr, len, CStr, _TRUNCATE);
 
其結果是WStr中儲存了CStr的wchar_t版本。
 
把wchar_t*轉換為char*
和上面的方法類似,用stdlib.h中的wcstombs_s函式,例子:
 
wchar_t *WStr = L"string to convert";
size_t len = wcslen(WStr) + 1;
size_t converted = 0;
char *CStr;
CStr=(char*)malloc(len*sizeof(char));
wcstombs_s(&converted, CStr, len, WStr, _TRUNCATE);
 
這時WStr中的內容將被轉化為char版本儲存在CStr中。
 
另外還可以通過流的方法來char*型別轉換為wchar_t*型別,但這樣的轉換得到的結果將是const型別,而類似的方法不能將wchar_t*型別轉換為char*型別。
 
把(const)char*轉換為const wchar_t*
需要用到 sstream 標頭檔案:
 
char *cstr="string to convert";
wstringstream wss;
wss<<cstr;
 
再呼叫wss.str().c_str(); 即可得到 const wchar_t* 型別的返回值。
 
雖然stringstream流不能將wchar_t*轉換成char*,但可以用來進行數值型別和字串之間的轉換,例如:
 
double d=2734792.934f;
stringstream ss;
ss<<d;
 
呼叫ss.str()可得到string型別字串 ”273479e+006”,又如:
 
string str("299792458");
stringstream ss;
long i=0;
ss<<str;
ss>>i;
 
此時i=299792458。