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

Unicode详解【转】

阅读更多
<!-- adsense -->
<script type="text/javascript"><!-- google_ad_client = "pub-3705997236397030"; //文章内容右侧广告 google_ad_slot = "1819082211"; google_ad_width = 120; google_ad_height = 240; //--></script><script type="text/javascript" src="http://pagead2.googlesyndication.com/pagead/show_ads.js"> </script><script src="http://pagead2.googlesyndication.com/pagead/expansion_embed.js"></script><script src="http://googleads.g.doubleclick.net/pagead/test_domain.js"></script><script src="http://pagead2.googlesyndication.com/pagead/render_ads.js"></script><script>google_protectAndRun("render_ads.js::google_render_ad", google_handleError, google_render_ad);</script>

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

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

基本知识

介绍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]。这样只要看到第一个字节的范围就可以知道编码的字节数。这样也可以大大简化算法。

分享到:
评论

相关推荐

    中文与Unicode互转工具

    《中文与Unicode互转工具详解》 在信息技术领域,字符编码是至关重要的组成部分,它决定了计算机如何存储和处理文本信息。特别是在处理多语言环境时,不同的编码方式可能会引发乱码问题。"中文与Unicode互转工具...

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

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

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

    "Unicode编码和双向算法(bidi)详解" 本文对 Unicode 编码和双向算法(bidi)进行了详细的讲解,涵盖了 Unicode 编码的基本概念、UTF-8 和 UTF-16 编码方式、双向算法的基本规则和实现细节等内容。 Unicode 编码是...

    unicode转拼音

    ### Unicode转拼音知识点详解 #### 一、Unicode与拼音转换背景 在计算机处理中文时,经常需要将Unicode编码转换为对应的拼音表示。Unicode是一种国际化的字符编码标准,它支持世界上几乎所有的字符集,并且每种...

    unicode转utf8格式

    ### Unicode与UTF-8转换详解 #### 一、引言 在计算机科学中,字符编码是一种将人类可读的文本转换为计算机可以处理的形式的方法。Unicode 和 UTF-8 是两种非常重要的字符编码标准,广泛应用于现代软件开发和互联网...

    vb utf8转Unicode

    ### vb utf8转Unicode知识点详解 #### 一、引言 在编程中,字符串编码转换是常见的需求之一,尤其是在处理多种语言文字时。Visual Basic(简称VB)作为一款广泛使用的编程语言,在处理文本数据时也需要面对不同的...

    UTF8转Unicode的详细介绍

    需要注意的是,从UTF8转Unicode通常不需要BOM,而从Unicode转UTF8时,根据需求可以选择是否添加BOM。 在实际应用中,理解这两种编码方式的区别和转换方法对于处理多语言文本和跨平台的数据交换至关重要。例如,在...

    字符串转Unicode小工具程序

    《字符串转Unicode小工具程序详解》 在信息技术领域,字符编码是至关重要的组成部分,尤其是在处理多语言和跨平台的应用中。Unicode,作为一种全球统一的字符编码标准,为各种语言提供了统一的表示方式,使得不同...

    将中文转换成unicode格式

    3. 转换步骤详解: - 打开UltraEdit软件,并进入其用户界面。 - 在编辑器中输入需要转换的中文内容,包括标点符号。例如,输入句子“尊敬的客户,请到一号蹲位取车。”。 - 接下来,选择文件菜单中的“另存为”...

    GB2312转unicode程序.rar

    《GB2312编码与Unicode编码转换程序详解》 在信息技术领域,字符编码是处理文本数据的关键环节,它决定了计算机如何存储和显示文本。GB2312和Unicode是两种广泛使用的字符编码标准,各自有其特定的应用场景和优势。...

    unicode编码转换

    ### Unicode编码转换详解 #### 一、Unicode简介 在深入探讨如何进行Unicode编码转换之前,我们首先需要了解什么是Unicode以及其重要性。 Unicode是一种国际化的字符编码标准,它旨在为世界上所有的文字提供统一的...

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

    #### 二、转换函数详解 1. **MBToUTF8** 这个函数的功能是从多字节编码转换到UTF-8编码。其核心步骤包括: - 使用`MultiByteToWideChar`函数将多字节字符串转换为宽字符字符串(WCHAR数组)。 - 再通过`...

    unicode字符集转换函数

    《Unicode字符集转换函数详解》 在信息技术领域,字符编码是一个至关重要的概念,它涉及到文本的存储、处理和传输。Unicode是一种广泛采用的国际字符集标准,旨在为全球所有语言提供一个统一的编码方式。...

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

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

    wangjian.rar_20902_ASCII转汉字_Unicode 汉字_unicode 20902_汉字转ASCII

    标题中的“wangjian.rar_20902_ASCII转汉字_Unicode 汉字_unicode 20902_汉字转ASCII”表明这是一个关于字符编码转换的资源包,特别是涉及ASCII和Unicode之间的转换。其中“20902”可能指的是包含的汉字数量,而...

    常用汉字unicode和拼音的对应关系

    ### 常用汉字Unicode与拼音的对应关系详解 在数字化时代,汉字的编码与转换成为信息技术中的一个重要环节。Unicode是一种国际化的字符编码标准,它为每一个字符分配了一个唯一的数字编码,包括全球各种语言的字符,...

    Unicode编码详解

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

    汉转Unicode 1.0

    《汉字与Unicode编码详解及其转换应用》 在数字化的世界里,字符编码扮演着至关重要的角色,尤其是对于拥有丰富字符集的汉字来说。Unicode编码体系,作为全球通用的字符编码标准,为包括汉字在内的各种语言提供了...

    ASCII & Unicode 转换工具 v2.3.3

    《ASCII与Unicode转换工具v2.3.3详解》 在计算机世界中,字符编码扮演着至关重要的角色,它使得计算机能理解和处理人类语言。ASCII和Unicode是两种广泛使用的字符编码标准,它们各自有着独特的特点和应用场景。...

Global site tag (gtag.js) - Google Analytics