`

Unicode详解

    博客分类:
  • CMPP
 
阅读更多

 

最近一直在忙点私活,又好久没写blog了,再不写点的话二月份就又要以单篇文章结束了。 前一阵子一直在研究Unicode,索性把研究结果介绍一下吧。

可能大家都听说过 Unicode、UCS-2、UTF-8 等等词汇,但它们具体是什么意思, 是什么原理,之间有什么关系,恐怕就很少有人明白了。 下面就分别介绍一下它们。


基本知识

介绍Unicode之前,首先要讲解一些基础知识。虽然跟Unicode没有直接的关系, 但想弄明白Unicode,没这些还真不行。

字节和字符的区别

咦,字节和字符能有什么区别啊?不都是一样的吗?完全正确,但只是在古老的DOS时代。 当Unicode出现后,字节和字符就不一样了。

字节(octet)是一个八位的存储单元,取值范围一定是0~255。而字符(character,或者word) 为语言意义上的符号,范围就不一定了。例如在UCS-2中定义的字符范围为0~65535, 它的一个字符占用两个字节。

Big Endian和Little Endian

上面提到了一个字符可能占用多个字节,那么这多个字节在计算机中如何存储呢? 比如字符0xabcd,它的存储格式到底是 AB CD,还是 CD AB 呢?

实际上两者都有可能,并分别有不同的名字。如果存储为 AB CD,则称为Big Endian; 如果存储为 CD AB,则称为Little Endian

具体来说,以下这种存储格式为Big Endian,因为值(0xabcd)的高位(0xab)存储在前面:

地址
0x00000000 AB
0x00000001 CD

相反,以下这种存储格式为Little Endian:

地址
0x00000000 CD
0x00000001 AB

UCS-2和UCS-4

Unicode是为整合全世界的所有语言文字而诞生的。任何文字在Unicode中都对应一个值, 这个值称为代码点(code point)。代码点的值通常写成 U+ABCD 的格式。 而文字和代码点之间的对应关系就是UCS-2(Universal Character Set coded in 2 octets)。 顾名思义,UCS-2是用两个字节来表示代码点,其取值范围为 U+0000~U+FFFF。

为了能表示更多的文字,人们又提出了UCS-4,即用四个字节表示代码点。 它的范围为 U+00000000~U+7FFFFFFF,其中 U+00000000~U+0000FFFF和UCS-2是一样的。

要注意,UCS-2和UCS-4只规定了代码点和文字之间的对应关系,并没有规定代码点在计算机中如何存储。 规定存储方式的称为UTF(Unicode Transformation Format),其中应用较多的就是UTF-16和UTF-8了。

UTF-16和UTF-32

UTF-16

UTF-16由RFC2781规定,它使用两个字节来表示一个代码点。

不难猜到,UTF-16是完全对应于UCS-2的,即把UCS-2规定的代码点通过Big Endian或Little Endian方式 直接保存下来。UTF-16包括三种:UTF-16,UTF-16BE(Big Endian),UTF-16LE(Little Endian)。

UTF-16BE和UTF-16LE不难理解,而UTF-16就需要通过在文件开头以名为BOM(Byte Order Mark)的字符 来表明文件是Big Endian还是Little Endian。BOM为U+FEFF这个字符。

其实BOM是个小聪明的想法。由于UCS-2没有定义U+FFFE, 因此只要出现 FF FE 或者 FE FF 这样的字节序列,就可以认为它是U+FEFF, 并且可以判断出是Big Endian还是Little Endian。

举个例子。“ABC”这三个字符用各种方式编码后的结果如下:

UTF-16BE 00 41 00 42 00 43
UTF-16LE 41 00 42 00 43 00
UTF-16(Big Endian) FE FF 00 41 00 42 00 43
UTF-16(Little Endian) FF FE 41 00 42 00 43 00
UTF-16(不带BOM) 00 41 00 42 00 43

Windows平台下默认的Unicode编码为Little Endian的UTF-16(即上述的 FF FE 41 00 42 00 43 00)。 你可以打开记事本,写上ABC,然后保存,再用二进制编辑器看看它的编码结果。

notepad-encode.png

另外,UTF-16还能表示一部分的UCS-4代码点——U+10000~U+10FFFF。 表示算法比较复杂,简单说明如下:

  1. 从代码点U中减去0x10000,得到U'。这样U+10000~U+10FFFF就变成了 0x00000~0xFFFFF。
  2. 用20位二进制数表示U'。 U'=yyyyyyyyyyxxxxxxxxxx
  3. 将前10位和后10位用W1和W2表示,W1=110110yyyyyyyyyy,W2=110111xxxxxxxxxx,则 W1 = D800~DBFF,W2 = DC00~DFFF。

例如,U+12345表示为 D8 08 DF 45(UTF-16BE),或者08 D8 45 DF(UTF-16LE)。

但是由于这种算法的存在,造成UCS-2中的 U+D800~U+DFFF 变成了无定义的字符。

UTF-32

UTF-32用四个字节表示代码点,这样就可以完全表示UCS-4的所有代码点,而无需像UTF-16那样使用复杂的算法。 与UTF-16类似,UTF-32也包括UTF-32、UTF-32BE、UTF-32LE三种编码,UTF-32也同样需要BOM字符。 仅用'ABC'举例:

UTF-32BE 00 00 00 41 00 00 00 42 00 00 00 43
UTF-32LE 41 00 00 00 42 00 00 00 43 00 00 00
UTF-32(Big Endian) 00 00 FE FF 00 00 00 41 00 00 00 42 00 00 00 43
UTF-32(Little Endian) FF FE 00 00 41 00 00 00 42 00 00 00 43 00 00 00
UTF-32(不带BOM) 00 00 00 41 00 00 00 42 00 00 00 43

UTF-8

UTF-16和UTF-32的一个缺点就是它们固定使用两个或四个字节, 这样在表示纯ASCII文件时会有很多00字节,造成浪费。 而RFC3629定义的UTF-8则解决了这个问题。

UTF-8用1~4个字节来表示代码点。表示方式如下:

UCS-2 (UCS-4) 位序列 第一字节 第二字节 第三字节 第四字节
U+0000 .. U+007F 00000000-0xxxxxxx 0xxxxxxx      
U+0080 .. U+07FF 00000xxx-xxyyyyyy 110xxxxx 10yyyyyy    
U+0800 .. U+FFFF xxxxyyyy-yyzzzzzz 1110xxxx 10yyyyyy 10zzzzzz  
U+10000..U+10FFFF 00000000-000wwwxx-
xxxxyyyy-yyzzzzzzz
11110www 10xxxxxx 10yyyyyy 10zzzzzz

可见,ASCII字符(U+0000~U+007F)部分完全使用一个字节,避免了存储空间的浪费。 而且UTF-8不再需要BOM字节。

另外,从上表中可以看出,单字节编码的第一字节为[00-7F],双字节编码的第一字节为[C2-DF], 三字节编码的第一字节为[E0-EF]。这样只要看到第一个字节的范围就可以知道编码的字节数。 这样也可以大大简化算法。

 

分享到:
评论

相关推荐

    Unicode详解(UTF-8,UTF16,UCS)

    Unicode详解(UTF-8,UTF16,UCS)

    Python中的字符串操作和编码Unicode详解

    ### Python中的字符串操作和编码Unicode详解 #### 一、引言 在Python编程语言中,字符串作为一种基础且常用的数据类型,具有丰富的操作方法和特性。其中,字符串的编码问题尤为重要,尤其是在处理国际化文本数据时...

    Delphi and Unicode说明

    ### Delphi与Unicode详解 #### 引言:Delphi 2009与Unicode Delphi 2009的一项重要新特性是它对Unicode字符集的全面支持。虽然仅针对英语编写的Delphi应用程序(基于26个字母的字母表)在以前的工作表现已经非常...

    Unicode.Explained.2006

    **Unicode详解2006** Unicode,全称统一码或万国码,是计算机科学领域里的一项标准,用于表示各种语言的文字。这个标准定义了一种字符集,旨在涵盖全球几乎所有的文字系统,包括拉丁字母、希腊字母、汉字、阿拉伯...

    浅谈文字编码和Unicode.pdf

    ### 文字编码与Unicode详解 #### 一、引言 随着信息技术的发展,文字编码成为连接计算机与人类语言的重要桥梁。不同的编码方式不仅关乎文本的存储和传输效率,更直接影响到跨语言、跨文化的交流能力。Unicode作为...

    Unicode explain

    **Unicode详解** Unicode是一种字符编码标准,旨在统一全球所有语言的字符表示,使其在不同的计算机系统间能够无缝传输和处理文本。它包含了世界上几乎所有的文字系统,从拉丁字母到汉字,再到阿拉伯文和印度文等。...

    keyboard unicode

    ### 键盘Unicode详解:以斯洛伐克键盘布局为例 在探讨键盘Unicode之前,我们首先需要理解何为Unicode以及其在键盘布局中的作用。Unicode是一种国际标准编码方案,旨在支持世界上所有书写系统的字符,包括拉丁字母、...

    双字节 多字节 宽字节 Unicode

    ### 双字节、多字节、宽字节与Unicode详解 #### 一、概述 在计算机科学领域,字符编码是将字符集映射到二进制数据的一种方式,以便于计算机存储和处理文本信息。本文将重点介绍双字节、多字节、宽字节与Unicode...

    unicode说明g

    《Unicode详解——ASCII与Unicode的差异与应用》 在信息技术领域,字符编码扮演着至关重要的角色,它使得计算机能够识别、存储和处理文本。Unicode和ASCII是两种广泛使用的字符编码系统,它们各自有其特点和应用...

    Unicode编码和双向算法(bidi)详解.pdf

    Unicode编码作为全球性的字符编码标准,其核心价值在于为每一种语言中的每一个字符提供一个统一且唯一的二进制编码。这种编码方式的出现,使得原本难以跨越的语言和平台障碍得到了有效解决,让计算机能够无歧义地...

    Unicode编码详解

    Unicode编码详解 Unicode编码是一种国际标准,用于在各种操作系统、编程语言、硬件设备间统一表示全球文字。这个标准由Unicode联盟制定,旨在解决早期字符编码系统(如ASCII和ISO 8859)无法覆盖所有语言的问题。...

    Windows核心编

    ### Windows核心编程之Unicode详解 #### 一、引言 随着全球化的推进,软件开发不再局限于单一市场,而是面向世界范围内的用户。Windows操作系统作为全球广泛使用的平台之一,其对多语言和多字符集的支持变得至关...

    unicode编码方式详解

    Unicode 编码方式详解 Unicode 编码方式是计算机程序或数据文件中文本存储和表示的一种方式。文本是由字符组成的,字符可以是字母、数字、句点、连字号、标点符号和数学符号等。为了在计算机中存储和表示这些字符,...

    文字各种编码的介绍与实现

    Unicode详解 #### 2.1 Unicode编码原理 - **基本原理**:Unicode为每一个字符分配一个唯一的数字,称为码点(Code Point)。这些码点被组织成一系列平面(Plane),每个平面包含65536(2^16)个码点。 - **平面与码点**...

    Unicode标准版本5.0The Unicode Standard, Version 5.0

    ### Unicode标准版本5.0详解 #### 一、Unicode简介 Unicode是一种国际化的字符编码标准,旨在为世界上所有书面语言的字符提供统一的编码方案。它不仅包括了拉丁字母、希腊字母、西里尔字母等常见字符集,还涵盖了...

    最全的unicode 汉字编码表

    #### Unicode汉字编码表详解 根据提供的文件内容,我们了解到该文件包含了一段特殊的Unicode编码表示例,范围为`4e00-9fa5`,这部分是Unicode标准中用于表示常用汉字的部分。 ##### 字符范围解析 - **起始码位**...

    Unicode字符集 V6.1

    ### Unicode字符集 V6.1 知识点详解 #### 一、Unicode标准简介 Unicode是一种全球性的字符编码标准,旨在提供一个通用的字符集,支持世界上几乎所有语言的文字表示。Unicode标准由Unicode联盟制定和维护,它定义了...

    js中unicode转码方法详解.docx

    JavaScript(简称JS)是一种广泛用于前端开发的轻量级脚本语言,它处理字符串时经常需要进行Unicode编码和解码。Unicode是一种国际标准,旨在表示世界上几乎所有的字符和符号,确保在不同系统间数据的兼容性。在JS中...

    GB与Unicode转换

    ### GB与Unicode转换详解 #### 一、引言 随着信息技术的发展,文本数据处理成为软件开发中的重要组成部分。为了确保不同编码系统之间的兼容性,掌握字符集间的转换机制显得尤为重要。本文将详细介绍GB码(国标码)...

Global site tag (gtag.js) - Google Analytics