对于需要字符串处理的WinAPI,系统大多会同时提供ANSI和Unicode版本的函数(有少数例外),分别以A和W结尾(如MessageBoxA和MessageBoxW),在windows.h及其包含文件中,按照是否定义UNICODE宏,将MessageBox转义为MessageBoxA或MessageBoxW。
由于目前基于NT内核的Windows内部是以Unicode作为系统编码的,因此ANSI版本的函数只是将字符串转码,然后调用Unicode版本的WinAPI,调用完毕之后,返回的Unicode字符串再转成ANSI编码。
ANSI版本的WinAPI,以gdi32.dll中的GetGlyphOutlineA为例,在调用对应Unicode版本的函数前,要调用三个函数将ANSI字符串转换为Unicode字符串:GdiGetCodePage、IsDBCSLeadByteEx和MultiByteToWideChar。其中IsDBCSLeadByteEx判断一个字节是否为DBCS(双字节字符集)的前导字节,然而54936(GB18030)或65001(UTF-8)代码页并不是一个DBCS,没有定义LeadByte的范围,因此显然不能用作默认的ANSI代码页。
比如:
<code class="lang-cpp">#include "stdafx.h"
#include <windows.h>
#include <stdio.h>
int _tmain(int argc, _TCHAR* argv[])
{
printf("是否前导字节(936):%d\n", IsDBCSLeadByteEx(936, '\xb1'));
printf("是否前导字节(950):%d\n", IsDBCSLeadByteEx(950, '\xb1'));
printf("是否前导字节(54936):%d\n", IsDBCSLeadByteEx(54936, '\xb1'));
printf("是否前导字节(65001):%d\n", IsDBCSLeadByteEx(65001, '\xb1'));
return 0;
}</stdio.h></windows.h></code>
输出:
但是WinAPI是可以用作控制台代码页的,方法很简单,在控制台中输入chcp 54936和chcp 65001,并把字体改成Lucida Console或Consolas即可,不能用点阵字体,否则遇到中文会报错。
参考资料:
XXXXXXXXXXXXXXXXXXXXXX/XXXXXXXXml
-