分享

关于wchar的两个常用函数wcstombs和WideCharToMultiByte

 邢孟哲图书馆 2014-06-05
宽字符已经困扰我很久了,以前我都是设置项目的属性把它改为多字节,不用UNICODE。不过现在又遇到宽字节的问题,没有办法,只有硬着头皮学学:
看了之后我才发现原来有wcsXXX的函数专门处理宽字节的,就是strXXX一样好使。呵呵,我不再惧怕了,就试着自己写了一下,还是学了蛮多东西的:
1.有wcsXXX的函数和strXXX的函数对应处理宽字节,wcslen就是求长度的,wcscmp就是比较两个字符串的。
2.输出也有相关的操作,wprintf(L”%s%s”);这样的操作,对文件也可以用fwprintf函数来输出。不过我发现貌似cout << wchar;不成功。也发现了一个问题,就是我输出”相等”这样一个字符串的时候,发现居然输出不正确,无论是控制台和文件都有错误。可见,这个还是有点小问题的。输出其他的例如”12345”等都是正常的。哎,这个函数并不可靠啊。
3.宽字节和普通串的转换问题,学了两个函数,一个是:
wcstombs(char* strDes, const wchar*, size_t nMax);这个函数的作用是把wchar转换为char。
char* strDes 为保存转换后的普通字符串,wchar* 要被转换的宽字符串。转换的最大长度。这里的长度是转换的个数,而不是字节长度。
mbstowcs() 就是一个相反的过程了,参数就不说了。

另一套转换的函数是:
int WideCharToMultiByte(
  UINT CodePage, 
  DWORD dwFlags, 
  LPCWSTR lpWideCharStr,
  int cchWideChar, 
  LPSTR lpMultiByteStr, 
  int cbMultiByte,
  LPCSTR lpDefaultChar,    
  LPBOOL lpUsedDefaultChar
);
他的参数很多,上面的连接有介绍,这里就不怎么细说了。
第一个是编码的方式,我一般用CP_ACP。第二个是转换标志,MSDN上说什么都不设置更快,然后我就什么都不管了就用NULL了。具体作用不知道,等遇到了再学。第三个参数就是被转换的字符串,第四个参数是该字符串的长度,-1表示自动算长度,如果是手动给出,一定要把最后的终结符长度也算上。我觉得还是-1来的实际。第五个参数就是保存转换串的指针,第六个参数就是保存串的长度,这里是单位字符的个数。如果转换的时候没有终结符,那么结果也没有终结符,要注意下。最后两个参数就是默认的填充字符和是否使用了默认填充字符,我一般就用NULL代替。

普通串转宽字节也是类似。

这里有几个注意的,一定要保证空间足够。还有就是那个长度是单位字符个数,而不是字节数,在转换时,推荐被转换的字符串长度设置为-1,因为这样他会自动算出终结符结束。返回值也是转换的单位字符个数。例如”相等”有普通串转换为宽字节串,返回结果是3,(有终结符),而反过来就是5。如果返回时0 说明转换失败。

  1. //mbstowcs()函数和wcstombs()函数,选自CRT库。平台无关,但需设定local。不设定Local,中文会乱码  
  2. bool UnicodeToAnsi(wchar_t* lpString,char* szAnsi)//(in,out)  
  3. {  
  4.     size_t nLen=wcslen(lpString)*2,nReturnLen;//wcslen <string.h>   
  5.     char* lpszBuf=(char*)malloc(nLen);//malloc <stdlib.h> and <malloc.h>   
  6.     nReturnLen=wcstombs(szAnsi,lpString,nLen);//wcstombs <stdlib.h>   
  7.     if(nReturnLen<0)  
  8.     {  
  9.         free(lpszBuf);  
  10.         return false;  
  11.     }  
  12.     strcpy(szAnsi,lpszBuf);  
  13.     return true;  
  14. }  
  1. bool UnicodeToAnsi(wchar_t* lpString,char* szAnsi)//(in,out)  
  2. {  
  3.     char* pWideCharStr;  
  4.     int nLenOfWideChar,nReturnlen;  
  5.     nLenOfWideChar=WideCharToMultiByte(CP_ACP,0,lpString,-1,NULL,0,NULL,NULL);  
  6.     //Header: Declared in Winnls.h; include Windows.h.  
  7.     if(!nLenOfWideChar)  
  8.         return false;  
  9.     pWideCharStr=(char*)HeapAlloc(GetProcessHeap(),HEAP_ZERO_MEMORY,nLenOfWideChar+1);  
  10.     nReturnlen=WideCharToMultiByte(CP_ACP,0,lpString,-1,pWideCharStr,nLenOfWideChar,NULL,NULL);  
  11.     if(!nReturnlen)  
  12.     {  
  13.         HeapFree(GetProcessHeap(),0,pWideCharStr);  
  14.         return false;  
  15.     }  
  16.     strcpy(szAnsi,pWideCharStr);  
  17.     HeapFree(GetProcessHeap(),0,pWideCharStr);  
  18.     return true;  
  19. }  

  1. bool AnsiToUnicode(char* lpString,wchar_t* szUnicode)//(in,out)  
  2. {  
  3.     wchar_t* pWideCharStr;  
  4.     int nLenOfWideChar,nReturnlen;  
  5.     nLenOfWideChar=MultiByteToWideChar(CP_ACP,0,lpString,-1,NULL,0);  
  6.     //Header: Declared in Winnls.h; include Windows.h.  
  7.     if(!nLenOfWideChar)  
  8.         return false;  
  9.     pWideCharStr=(wchar_t*)HeapAlloc(GetProcessHeap(),HEAP_ZERO_MEMORY,nLenOfWideChar*2);  
  10.     nReturnlen=MultiByteToWideChar(CP_ACP,0,lpString,-1,pWideCharStr,nLenOfWideChar);  
  11.     if(!nReturnlen)  
  12.     {  
  13.         HeapFree(GetProcessHeap(),0,pWideCharStr);  
  14.         return false;  
  15.     }  
  16.     wcscpy(szUnicode,pWideCharStr);  
  17.     HeapFree(GetProcessHeap(),0,pWideCharStr);  
  18.     return true;  
  19. }  



    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多