目录

第1章小心Windows7的UTF-8代码页    1

1.1 UTF-16与UTF-8相互转换    1

1.1.1 使用Windows API    1

1.1.2 自己编码    1

1.2 测试代码    4

1.3 测试结果    5

第1章小心Windows7的UTF-8代码页

1.1 UTF-16与UTF-8相互转换

发现Windows7的UTF-8代码页有问题的根源就在于UTF-16与UTF-8的相互转换。

1.1.1 使用Windows API

使用Windows API,可以实现UTF-16编码与UTF-8编码的相互转换,如下面的代码:

char a1[128];

wchar_t w = 0;

int n1 = 0;

wchar_t w1[128];

int m1 = 0;

n1 = WideCharToMultiByte(CP_UTF8,0,&w,1,a1,128,NULL,NULL);

m1 = MultiByteToWideChar(CP_UTF8,0,a1,n1,w1,128);

WideCharToMultiByte把UTF-16编码转换为UTF-8编码;

MultiByteToWideChar把UTF-8编码转换为UTF-16编码。

API函数WideCharToMultiByte和MultiByteToWideChar能够正常工作的前提是:Windows系统已经安装了UTF-8代码页。Windows XP、Windows 7默认已经安装了UTF-8代码页,因此不用太担心。Windows Mobile系统就有些麻烦了:有些Windows Mobile安装了UTF-8代码页,而有些没有。所以,为了程序在所有Mobile设备上正常运行,是不能使用WideCharToMultiByte和MultiByteToWideChar的。

1.1.2 自己编码

UTF-16与UTF-8的相互转换,不能使用WideCharToMultiByte和MultiByteToWideChar,怎么办?所幸的是:UTF-16与UTF-8的相互转换是有严格的转换关系的。本文不做这方面的论述,直接上VC++代码:

/*******************************************************************\

将 UTF16 字符串转换为 UTF8 字符串

pUTF16 [in] UTF16 字符串首地址

nLen16 [in] UTF16 字符串字符数,小于 0 表示以 \0 结尾

pUTF8 [out] UTF8 字符串首地址,可以为 NULL

返回:转换的 UTF8 字符个数

\*******************************************************************/

long UTF16to8(const wchar_t*pUTF16,long nLen16,char*pUTF8)

{

long nLen8 = 0;

if(pUTF16)

{

if(nLen16 < 0)

{

nLen16 = wcslen(pUTF16) + 1;

}

}

else

{

nLen16 = 0;

}

if(nLen16 > 0)

{

long i = 0;

wchar_t u = 0;

for(i = 0;i < nLen16;++i)

{

u = pUTF16[i];

if(u <= 0x7F)

{

if(pUTF8)

{

*pUTF8++ = (char)u;

}

++nLen8;

}

else if(u <= 0x7FF)

{

if(pUTF8)

{//110xxxxx 10xxxxxx

pUTF8[1] = (char)(u & 0x3F | 0x80); u >>= 6;

pUTF8[0] = (char)(u & 0x1F | 0xC0);

pUTF8 += 2;

}

nLen8 += 2;

}

else

{

if(pUTF8)

{//1110xxxx 10xxxxxx 10xxxxxx

pUTF8[2] = (char)(u & 0x3F | 0x80); u >>= 6;

pUTF8[1] = (char)(u & 0x3F | 0x80); u >>= 6;

pUTF8[0] = (char)(u & 0x0F | 0xE0);

pUTF8 += 3;

}

nLen8 += 3;

}

}

}

return nLen8;

}

/*******************************************************************\

将 UTF8 字符串转换为 UTF16 字符串

pUTF8 [in] UTF8 字符串首地址

nLen8 [in] UTF8 字符串字符数,小于 0 表示以 \0 结尾

pUTF16 [out] UTF16 字符串首地址,可以为 NULL

返回:转换的 UTF16 字符个数

\*******************************************************************/

long UTF8to16(const char*pUTF8,long nLen8,wchar_t*pUTF16)

{

long nLen16 = 0;

if(pUTF8)

{

if(nLen8 < 0)

{

nLen8 = strlen(pUTF8) + 1;

}

}

else

{

nLen8 = 0;

}

if(nLen8 > 0)

{

long i = 0;

unsigned char u = 0;

for(i = 0;i < nLen8;)

{

u = pUTF8[i];

if(u >= 0xE0)

{

if(pUTF16)

{

*pUTF16++ = (pUTF8[i] & 0x0f) << 12

| (pUTF8[i + 1] & 0x3f) << 6

| (pUTF8[i + 2] & 0x3f);

}

i += 3;

}

else if(u >= 0xC0)

{

if(pUTF16)

{

*pUTF16++ = (pUTF8[i] & 0x1f) << 6

| (pUTF8[i + 1] & 0x3f);

}

i += 2;

}

else

{

if(pUTF16)

{

*pUTF16++ = u;

}

++i;

}

++nLen16;

}

}

return nLen16;

}

1.2 测试代码

自己编写的代码在使用前,那是要测试的。下面就是测试代码。其实就是与WideCharToMultiByte和MultiByteToWideChar进行比较:

void Test()

{

char a1[128];

char a2[128];

wchar_t w;

int n1 = 0;

int n2 = 0;

wchar_t w1[128];

wchar_t w2[128];

int m1 = 0;

int m2 = 0;

for(long i = 0;i <= 0xFFFF;++i)

{

w = (wchar_t)i;

n1 = WideCharToMultiByte(CP_UTF8,0,&w,1,a1,128,NULL,NULL);

m1 = MultiByteToWideChar(CP_UTF8,0,a1,n1,w1,128);

if(m1 != 1 && w1[0] != w)

{//WideCharToMultiByte与MultiByteToWideChar相互转换有问题

TRACE(_T("WideCharToMultiByte,MultiByteToWideChar=%04X\n"),w);

}

n2 = UTF16to8(&w,1,a2);

m2 = UTF8to16(a2,n2,w2);

if(m2 != 1 && w2[0] != w)

{//UTF16to8与UTF8to16相互转换有问题

TRACE(_T("UTF16to8,UTF8to16=%04X\n"),w);

}

if(n1 != n2 || memcmp(a1,a2,n1))

{//UTF16to8与WideCharToMultiByte有出入

TRACE(_T("WideCharToMultiByte,UTF16to8=%04X\n"),w);

}

}

}

1.3 测试结果

测试结果是:

1、在Windows XP下一切正常;

2、在Windows 7(64位旗舰版)下出问题了:当w在0xD800~0xD98A之间时,UTF16to8与WideCharToMultiByte的转换结果不一致!

笔者相信自己的代码,同时又有Windows XP的证明,最终得出的结论就是:Windows7的UTF-8代码页有问题!

还好,0xD800~0xD98A的395个字符并不是很常用,否则Windows7保存的UTF-8文件到了Windows XP或Linux下显示出乱码,问题就严重了。

解决方案:UTF-16与UTF-8编码的相互转换,还是自己编码实现吧。

小心Windows7的UTF-8代码页的更多相关文章

  1. CodePage------Encoding 类支持的编码以及与这些编码关联的代码页(CodePage)

    Encoding 类 .NET Framework 4  表示字符编码. 继承层次结构 System.Object  System.Text.Encoding    System.Text.ASCII ...

  2. 刨根究底字符编码之七——ANSI编码与代码页(Code Page)

    ANSI编码与代码页(Code Page) 一.ANSI编码 1. 如前所述,在全世界所有国家和民族的文字符号统一编码的Unicode编码方案问世之前,各个国家.民族为了用计算机记录并显示自己的字符, ...

  3. CMD代码页

    不同字符编码在CMD模式下会出现乱码,需要使用 chcp 代码页 命令来更改代码页显示正常. UTF-8  65001 简体中文 936 437          美国 850          多语 ...

  4. 解决“在多字节的目标代码页中,没有此Unicode字符可以映射到的字符”

    今天在处理Google网站管理员中的500错误时发现这样一些URL: http://www.cnblogs.com/Garnai/tag/3D%3F%96%CA/ http://www.cnblogs ...

  5. Windows代码页、区域

    目录 第1章代码页    1 1 代码页    1 1.1 单字节字符集    1 1.2 双字节字符集    1 1.3 多字节字符集    1 1.4 ANSI代码页    2 2 枚举代码页   ...

  6. 更改cmd代码页,修正语言显示

    1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 rem 英文 chcp 437   rem 日文 chcp 932   rem 简体中文 chcp 936   re ...

  7. warning C4819: 该文件包含不能在当前代码页(936)中表示的字符。请将该文件保存为 Unicode 格式以防止数据丢失

    bug来源: 一直在看sift然后就手贱的给 opencv源码做注释,如果在vs里面打开会一直相安无事,但是问题出在我用了notepad++. 这样就报了标题的错误. 因为notepad++会以uni ...

  8. Visual Studio在页面按F7不能跳转至cs代码页的解决方法

    检查页面Page设置内的CodeBehind属性,看是否与代码页的文件名相同,不同则改正,问题得以解决.

  9. warning: C4819: 该文件包含不能在当前代码页(936)中表示的字符。请将该文件保存为 Unicode 格式以防止数据丢失

    ------问题-------------------- Qt项目使用 VC++ 编译器出现此错误. warning: C4819: 该文件包含不能在当前代码页(936)中表示的字符.请将该文件保存为 ...

随机推荐

  1. 2016年6月26日 星期日 --出埃及记 Exodus 14:23

    2016年6月26日 星期日 --出埃及记 Exodus 14:23 The Egyptians pursued them, and all Pharaoh's horses and chariots ...

  2. C语言中的const

    今天探讨const,首先来说是将变量常量化.为什么要将变量常量化,原因有诸多好处有诸多.比如可以使数据更加安全不会被修改! 但是这个词有几个点要注意,那就是他究竟修饰了谁? 1.const int a ...

  3. 使用js实现移动设备访问跳转到指定目录

    最近最项目的时候总会同时做pc站点跟手机站点,当手机访问的时候默认是看到pc站点的,需要在url上加上/mobile才能正常访问,这段代码是我同事分享给我的,还是蛮实用的. CODE function ...

  4. UICollectionView 使用

    /** 初始化UICollectionView */ UICollectionViewFlowLayout *flowLayout=[[UICollectionViewFlowLayout alloc ...

  5. 不同分辨率的LCM进行兼容

    1.读取不同LCM ID的时序不同,如何处理? 2.开机后android会自动resize图片资源的大小,但开机logo无法自行resize,如何操作?保证识别到不同分辨率的LCM后开机logo可以正 ...

  6. 2.建立exception包,建立Bank类,类中有变量double balance表示存款,Bank类的构造方法能增加存款,Bank类中有取款的发方法withDrawal(double dAmount),当取款的数额大于存款时,抛出InsufficientFundsException,取款数额为负数,抛出NagativeFundsException,

    public class Bank { Double qian=0.0; double newBank(double a) { qian=qian+a; return qian; } double w ...

  7. CodeForces 148B Escape

    Escape Time Limit:2000MS     Memory Limit:262144KB     64bit IO Format:%I64d & %I64u Submit Stat ...

  8. thinkphp的自动完成功能说明

    手册里有一句话很关键: 自动完成是ThinkPHP提供用来完成数据自动处理和过滤的方法,使用create方法创建数据对象的时候会自动完成数据处理. 这句话说明自动完成发生的时间是create()组建数 ...

  9. CUBRID学习笔记 43 insert into

    cubrid的中sql查询语法insert into ------ 官方文档是英文的,看不明白可以参看ocracle的同类函数说明.很多都是一样的. INSERT INTO a_tbl1(id) VA ...

  10. kakfa的常用命令总结

    Kafka的版本间差异较大,下面是0.8.2.1的操作方法 首先cd到kafaka的bin目录下;   #step1启动zookeeper服务 nohup bin/zookeeper-server-s ...