`
daragon
  • 浏览: 48944 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

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

阅读更多

  转载于http://tech.idv2.com/2008/02/21/unicode-intro/  原作者charlee

Unicode详解

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

<!-- end Pukiwiki generated code--><!-- begin Pukiwiki generated code-->


  • 基本知识
    • 字节和字符的区别
    • Big Endian和Little Endian
  • UCS-2和UCS-4
  • UTF-16和UTF-32
    • UTF-16
    • UTF-32
  • 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+1FFFFF 00000000-000wwwxx-
xxxxyyyy-yyzzzzzzz
11110www 10xxxxxx 10yyyyyy 10zzzzzz

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

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

分享到:
评论
1 楼 windloverain 2011-05-18  
 

相关推荐

    多字节与UTF-8、Unicode之间的转换

    3. **Unicode**:Unicode是一种标准,旨在为世界上所有的字符提供统一且唯一的数字表示形式,通常以UCS-2或UCS-4的形式存储,分别使用2个或4个字节表示一个字符。这里我们关注的是UCS-2,即每个字符使用2个字节表示...

    unicode/UCS/UTF-8/Base64/ANSI等编码介绍

    《编码规则与用法详解:Unicode、UCS、UTF-8、Base64、ANSI》 在信息技术领域,字符编码是至关重要的,它决定了我们如何在计算机中存储、传输和显示文本。本文将详细介绍几种常见的编码方式,包括Unicode、UCS、UTF...

    详解utf8字符集

    ### 详解UTF8字符集 #### 一、引言 随着全球化的推进和技术的发展,不同语言和字符系统之间的兼容性和互通性变得尤为重要。Unicode作为一套国际标准字符编码方案,旨在解决这一问题。然而,Unicode本身的编码方式...

    UNICODE 基础知识.pdf

    Unicode及其不同的编码形式(如UTF-8、UTF-16和UTF-32)为全球范围内的文字表示提供了强大的支持。通过不断扩展编码范围和支持不同类型的编码方案,Unicode已经成为国际标准化组织认可的标准之一,广泛应用于互联网...

    常用编码详解 常用编码详解 常用编码详解 常用编码详解

    从UCS-4到UTF-8的转换规则是确定所需字节数,设置字节高位,然后填充UCS-4字符值的低位。这种转换使得UCS字符可以适应不同字节长度的编码格式。 总的来说,这些编码标准在现代信息技术中扮演着不可或缺的角色。UCS...

    UTF8基础知识

    UTF-16编码在内部处理和存储上可能比UTF-8更高效,但在网络传输时可能会占用更多空间。 #### 结论 UTF-8作为Unicode的编码方式之一,其优势在于对ASCII字符的高效表示和对全球所有语言的全面支持。理解UTF-8及其与...

    常用编码详解.docx

    UTF-8编码是UCS-2和UCS-4编码的变种,它使用了8比特字节的所有位,保持全部US-ASCII取值范围的性质:US-ASCII字符用一个8比特字节编码,采用通常的US-ASCII值,因此,在此值下的任何一个8比特位字节仅仅代表一个US-...

    Unicode编码

    - **UTF-16**:UTF-16是为了解决UCS-2的问题而设计的,它同样使用2个字节来表示BMP中的字符,但对于非BMP字符,则使用4个字节(即两个UTF-16码单元)来表示。 - **关系**:在处理BMP内的字符时,UCS-2和UTF-16是完全...

    GB2311 UNICODE字符集以及字符编码知识了解

    例如,将GB2312编码的数据转换为Unicode或UTF-8格式,或者将UTF-16的数据转换为UTF-8格式等。 #### 七、字节顺序标记(Byte Order Mark, BOM) 在处理Unicode编码时,还涉及到了字节序的问题。字节序是指在内存中...

    Java Unicode 和字符集

    随着Unicode字符集的不断扩大,UCS-2已经不足以覆盖所有字符,因此引入了更高级的编码方式——UTF-16。UTF-16仍然使用16位的基本单位,但对于超出基本多文种平面(Basic Multilingual Plane,BMP)的字符,则使用更...

    phpstorm语言包

    首先,Unicode16编码,也称为UCS-2,是一种固定宽度的16位编码,它可以表示Unicode字符集中大部分的基本多文种平面(BMP)字符。这种编码方式对中文的支持非常好,因为汉字主要位于BMP范围内。但是,Unicode16不支持...

    Unicode编码源代码

    - **UTF-8、UTF-16和UTF-32**: UTF-8是最常用的Unicode变体,它以可变长度的方式编码字符,从1到4个字节不等。UTF-16则通常用2个字节表示字符,对于UCS-2范围内的字符与UCS-2相同,而超出的字符使用4个字节。UTF-32...

    java字符集编码问题

    它有两种主要形式:UCS-2(Unicode Character Set - 2 bytes)和UCS-4(Unicode Character Set - 4 bytes)。UCS-2是最常用的,每个字符使用两个字节表示。Unicode不兼容其他编码标准,因为它采用定长编码,便于...

    python3编码问题汇总

    最初的Unicode版本(UCS-2)使用两个字节来表示字符,最多可以表示65536个不同的字符。随着语言的发展,后来又引入了UCS-4,使用四个字节来表示字符。 ##### 4.4 UTF-8编码 UTF-8(Unicode Transformation Format-8 ...

    Unicode编码方案.docx

    - **编码方式**:定义了如何将Unicode码点映射到二进制数据序列,常见的有UTF-8、UTF-16、UTF-32等。 - **实现方式**:指的是具体的数据结构和技术细节,用于存储和处理Unicode字符。 - **设计原则**:根据《The ...

    各种字符编码方式详解及由来

    一直对字符的各种编码方式懵懵懂懂,什么ANSI UNICODE UTF-8 GB2312 GBK DBCS UCS……是不是看的很晕,假如您细细的阅读本文你一定可以清晰的理解他们

Global site tag (gtag.js) - Google Analytics