`
carge
  • 浏览: 51912 次
最近访客 更多访客>>
文章分类
社区版块
存档分类
最新评论

字符编码和URL编码的学习

 
阅读更多

 

 

第一章 名词解释[1]

【注】以下内容来自我好朋友日志,请参阅参考文献[1]

   字、字符、字符集、字符集编码、字符集编码格式。下面我就尝试对这几个概念做个解释。

 

 1、字

     字是形而上的东西,也就是它没有具体的实物,它可能有音形义等外在表现。有些字没有音,如:“?”(“问号”只是“?”的名字,不是“?”的发音);有些字没有形,如:字符串结束符等不可显示字符,它们不可见,但有特殊的作用;有些字没义,这我就不能举例,只要一举例它就有义,但只要你不理它,所有的字都是没有义的,因为意由心生。

 

2、字符

     字符(Character)是各种文字和符号的总称,包括各国家文字、标点符号、图形符号、数字等[2]。

     表示字的符号就是字符(绝对是望文生意),即字的形。在生活中,我们常常把字符当成字,这是一种习惯,因为形而上的东西太难说清楚了,所以大家明白在说什么就够了,不用去深度挖掘它的概念,不然到最后怎么讲也讲不清了。

 

3、字符集

    字符集就是多个字符的集合。因此你可以随便抓几个字符凑在一起,然后说这就是一个字符集。但在计算机世界里为了交流的目的,一定要用和别人相同的或者兼容的字符集,不然在电脑A中的字符在电脑B中找不到,就会有字符显示的问题。因此字符集要有标准,或者说规范、协议、约定等。

    在国内常见的标准字符集有:ANSII、ISO-8859-1、GB2312、BIG5、CJK、GBK、GB18030、UCS、Unicode等。字符集之间有兼容性的问题,如果说字符集A兼容字符集B,那只有一种情况:A包含B中所有字符。在上面的字符当中,ASCII是最小的字符集,被其它字符集兼容。而UCS和Unicode几乎是最大的字符集,它们俩基本相互兼容,而其它所有字符集都是它们的子集。

 

4、字符集编码

    计算机要准确的处理各种字符集文字,需要进行字符编码,以便计算机能够识别和存储各种文字。在一个字符集当中,字符的一个排列就应该可以说是一种编码(我猜的,没有人告诉我怎么定义,估且先这么说吧),每个字符的编码就是它在排列中的序号。为了交流的目的,一定要和别人使用相同的或者兼容的字符集编码,因此上述常见的字符集也各有相应编码标准。因为人们在制定这些字符集,同时也制定了相应的编码,所以现在基本上可将上述字符集都看成有序集。字符集编码之间同样有兼容问题,且只有大的字符集的编码才能兼容小的字符集的编码。如果说字符集A的编码兼容字符集B的编码,那首先是A兼容B,其次是B中所有字符在A和B中都有相同的编码。

 

5、字符集编码格式

     对编码的表示方式就是编码格式。格式有两点作用,一是能够指出所表示的编码值;二是当不同字符的编码按这个格式表示成二进制编码并且连接成一个串时,它能够把这个串正确分开,还原成未连接前的状态。不同的格式也有兼容的问题,如果格式A兼容格式B,则按格式A的去看格式B的所有编码串,能解码得到相同的字符串。上述字符集的编码格式也有相应的规定。例如:ASCII字符集只有128个字符,所以只要7个比特就可以全部表示出来,因此ASCII格式用一个字节表示一个字符,最高位固定为0;ISO-8859-1有256个字符,因此ISO-8859-1格式用一个字节表示一个字符,而且兼容ASCII格式。其它的格式当中,GB2312格式兼容ASCII格式,GBK格式兼容GB2312格式,GB18030格式兼容GBK格式。还有一种使用较广的编码格式utf-8,它是Unicode字符集的编码格式,只兼容ASCII格式。国内常用GBK格式,因此utf-8格式和GBK格式的不兼容问题常给我们造成麻烦。幸好Unicode字符集兼容GBK字符集(但它们的编码不兼容),只要使用的字符不超过GBK字符集,我们就可以在两者之间做转换。

 

第二章 Unicode[3]

1、UCS(Universal Character Set, 通用字符集)[4]

    国际标准 ISO 10646 定义了通用字符集 (Universal Character Set, UCS). UCS 是所有其他字符集标准的一个超集. 它保证与其他字符集是双向兼容的. 就是说, 如果你将任何文本字符串翻译到 UCS格式, 然后再翻译回原编码, 你不会丢失任何信息.

  Unicode 是基于通用字符集(Universal Character Set)的标准来发展,并且同时也以书本的形式(The Unicode Standard,目前第五版由Addison-Wesley Professional出版。ISBN-10:0321480910)对外发表。

 

2、Unicode 的编码和实现

大概来说,Unicode 编码系统可分为编码方式和实现方式两个层次。

 

2.1 编码方式

  Unicode是国际组织制定的可以容纳世界上所有文字和符号的字符编码方案。Unicode用数字0-0x10FFFF来映射这些字符,最多可以容纳1114112个字符,或者说有1114112个码位。码位就是可以分配给字符的数字。UTF-8、UTF-16、UTF-32都是将数字转换到程序数据的编码方案。

  Unicode字符集可以简写为UCS(Unicode Character Set)。早期的Unicode标准有UCS-2、UCS-4的说法。UCS-2用两个字节编码,UCS-4用4个字节编码。UCS-4根据最高位为0的最高字节分成2^7=128个group。每个group再根据次高字节分为256个平面(plane)。每个平面根据第3个字节分为256行 (row),每行有256个码位(cell)。group 0的平面0被称作BMP(Basic Multilingual Plane)。将UCS-4的BMP去掉前面的两个零字节就得到了UCS-2。

  每个平面有2^16=65536个码位。Unicode计划使用了17个平面,一共有17*65536=1114112个码位。在Unicode 5.0.0版本中,已定义的码位只有238605个,分布在平面0、平面1、平面2、平面14、平面15、平面16。其中平面15和平面16上只是定义了两个各占65534个码位的专用区(Private Use Area),分别是0xF0000-0xFFFFD和0x100000-0x10FFFD。所谓专用区,就是保留给大家放自定义字符的区域,可以简写为PUA。

  平面0也有一个专用区:0xE000-0xF8FF,有6400个码位。平面0的0xD800-0xDFFF,共2048个码位,是一个被称作代理区(Surrogate)的特殊区域。代理区的目的用两个UTF-16字符表示BMP以外的字符。在介绍UTF-16编码时会介绍。

  如前所述在Unicode 5.0.0版本中,238605-65534*2-6400-2408=99089。余下的99089个已定义码位分布在平面0、平面1、平面2和平面14上,它们对应着Unicode目前定义的99089个字符,其中包括71226个汉字。平面0、平面1、平面2和平面14上分别定义了52080、3419、43253和337个字符。平面2的43253个字符都是汉字。平面0上定义了27973个汉字。

 

2.2 实现方式

  在Unicode中:汉字“字”对应的数字是23383。在Unicode中,我们有很多方式将数字23383表示成程序中的数据,包括:UTF-8、UTF-16、UTF-32。UTF是“UCS Transformation Format”的缩写,可以翻译成Unicode字符集转换格式,即怎样将Unicode定义的数字转换成程序数据。例如,“汉字”对应的数字是0x6c49和0x5b57,而编码的程序数据是:

  BYTE data_utf8[] = {0xE6, 0xB1, 0x89, 0xE5, 0xAD, 0x97}; // UTF-8编码
  WORD data_utf16[] = {0x6c49, 0x5b57}; // UTF-16编码
  DWORD data_utf32[] = {0x6c49, 0x5b57}; // UTF-32编码 

   这里用BYTE、WORD、DWORD分别表示无符号8位整数,无符号16位整数和无符号32位整数。UTF-8、UTF-16、UTF-32分别以BYTE、WORD、DWORD作为编码单位。“汉字”的UTF-8编码需要6个字节。“汉字”的UTF-16编码需要两个WORD,大小是4个字节。“汉字”的UTF-32编码需要两个DWORD,大小是8个字节。根据字节序的不同,UTF-16可以被实现为UTF-16LE或UTF-16BE,UTF-32可以被实现为UTF-32LE或UTF-32BE。下面介绍UTF-8、UTF-16、UTF-32、字节序和BOM。

 

  UTF-8

  UTF-8以字节为单位对Unicode进行编码。从Unicode到UTF-8的编码方式如下:

  Unicode编码(16进制) ║ UTF-8 字节流(二进制)

  000000 - 00007F ║ 0xxxxxxx

  000080 - 0007FF ║ 110xxxxx 10xxxxxx

  000800 - 00FFFF ║ 1110xxxx 10xxxxxx 10xxxxxx

  010000 - 10FFFF ║ 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx

  UTF-8的特点是对不同范围的字符使用不同长度的编码。对于0x00-0x7F之间的字符,UTF-8编码与ASCII编码完全相同。UTF-8编码的最大长度是4个字节。从上表可以看出,4字节模板有21个x,即可以容纳21位二进制数字。Unicode的最大码位0x10FFFF也只有21位。

  例1:“汉”字的Unicode编码是0x6C49。0x6C49在0x0800-0xFFFF之间,使用用3字节模板了:1110xxxx 10xxxxxx 10xxxxxx。将0x6C49写成二进制是:0110 1100 0100 1001, 用这个比特流依次代替模板中的x,得到:11100110 10110001 10001001,即E6 B1 89。

  例2:Unicode编码0x20C30在0x010000-0x10FFFF之间,使用用4字节模板了:11110xxx 10xxxxxx 10xxxxxx 10xxxxxx。将0x20C30写成21位二进制数字(不足21位就在前面补0):0 0010 0000 1100 0011 0000,用这个比特流依次代替模板中的x,得到:11110000 10100000 10110000 10110000,即F0 A0 B0 B0。

 

  UTF-16

  UTF-16编码以16位无符号整数为单位。我们把Unicode  

unicode

编码记作U。编码规则如下:

  如果U<0x10000,U的UTF-16编码就是U对应的16位无符号整数(为书写简便,下文将16位无符号整数记作WORD)。

  如果U≥0x10000,我们先计算U'=U-0x10000,然后将U'写成二进制形式:yyyy yyyy yyxx xxxx xxxx,U的UTF-16编码(二进制)就是:110110yyyyyyyyyy 110111xxxxxxxxxx。

  为什么U'可以被写成20个二进制位?Unicode的最大码位是0x10ffff,减去0x10000后,U'的最大值是0xfffff,所以肯定可以用20个二进制位表示。例如:Unicode编码0x20C30,减去0x10000后,得到0x10C30,写成二进制是:0001 0000 1100 0011 0000。用前10位依次替代模板中的y,用后10位依次替代模板中的x,就得到:1101100001000011 1101110000110000,即0xD843 0xDC30。

  按照上述规则,Unicode编码0x10000-0x10FFFF的UTF-16编码有两个WORD,第一个WORD的高6位是110110,第二个WORD的高6位是110111。可见,第一个WORD的取值范围(二进制)是11011000 00000000到11011011 11111111,即0xD800-0xDBFF。第二个WORD的取值范围(二进制)是11011100 00000000到11011111 11111111,即0xDC00-0xDFFF。

  为了将一个WORD的UTF-16编码与两个WORD的UTF-16编码区分开来,Unicode编码的设计者将0xD800-0xDFFF保留下来,并称为代理区(Surrogate):

  D800-DB7F ║ High Surrogates ║ 高位替代

  DB80-DBFF ║ High Private Use Surrogates ║ 高位专用替代

  DC00-DFFF ║ Low Surrogates ║ 低位替代

  高位替代就是指这个范围的码位是两个WORD的UTF-16编码的第一个WORD。低位替代就是指这个范围的码位是两个WORD的UTF-16编码的第二个WORD。那么,高位专用替代是什么意思?我们来解答这个问题,顺便看看怎么由UTF-16编码推导Unicode编码。

  如果一个字符的UTF-16编码的第一个WORD在0xDB80到0xDBFF之间,那么它的Unicode编码在什么范围内?我们知道第二个WORD的取值范围是0xDC00-0xDFFF,所以这个字符的UTF-16编码范围应该是0xDB80 0xDC00到0xDBFF 0xDFFF。我们将这个范围写成二进制:

  1101101110000000 11011100 00000000 - 1101101111111111 1101111111111111

  按照编码的相反步骤,取出高低WORD的后10位,并拼在一起,得到

  1110 0000 0000 0000 0000 - 1111 1111 1111 1111 1111  

XML

即0xe0000-0xfffff,按照编码的相反步骤再加上0x10000,得到0xf0000-0x10ffff。这就是UTF-16编码的第一个WORD在0xdb80到0xdbff之间的Unicode编码范围,即平面15和平面16。因为Unicode标准将平面15和平面16都作为专用区,所以0xDB80到0xDBFF之间的保留码位被称作高位专用替代。

 

  UTF-32

  UTF-32编码以32位无符号整数为单位。Unicode的UTF-32编码就是其对应的32位无符号整数。

  字节序

  根据字节序的不同,UTF-16可以被实现为UTF-16LE或UTF-16BE,UTF-32可以被实现为UTF-32LE或UTF-32BE。例如:

  Unicode编码 ║ UTF-16LE ║ UTF-16BE ║ UTF32-LE ║ UTF32-BE

  0x006C49 ║ 49 6C ║ 6C 49 ║ 49 6C 00 00 ║ 00 00 6C 49

  0x020C30 ║ 43 D8 30 DC ║ D8 43 DC 30 ║ 30 0C 02 00 ║ 00 02 0C 30

  那么,怎么判断字节流的字节序呢?Unicode标准建议用BOM(Byte Order Mark)来区分字节序,即在传输字节流前,先传输被作为BOM的字符"零宽无中断空格"。这个字符的编码是FEFF,而反过来的FFFE(UTF-16)和FFFE0000(UTF-32)在Unicode中都是未定义的码位,不应该出现在实际传输中。下表是各种UTF编码的BOM:

  UTF编码 ║ Byte Order Mark

  UTF-8 ║ EF BB BF

  UTF-16LE ║ FF FE

  UTF-16BE ║ FE FF

  UTF-32LE ║ FF FE 00 00

  UTF-32BE ║ 00 00 FE FF



 

 

[1] 油头饼日志,http://user.qzone.qq.com/948509263/blog/1283446231

[2]字符集, http://baike.baidu.com/view/51987.htm

[3]Unicode, http://baike.baidu.com/view/40801.htm

[4]UCS, http://baike.baidu.com/view/935284.htm

分享到:
评论

相关推荐

    java使用URLDecoder和URLEncoder对中文字符进行编码和解码

    这两个类位于`java.net`包下,可以帮助开发者进行字符串编码和解码,确保数据在网络传输过程中的正确性。 `URLEncoder`类提供了`encode`静态方法,用于将普通字符串编码成`application/x-www-form-urlencoded` MIME...

    java字符编码监听器

    Java字符编码监听器是Java Web开发中的一个重要概念,主要用于处理HTTP请求和响应中的字符编码问题。在Java Servlet规范中,提供了`SetCharacterEncodingFilter`这样的过滤器,用于确保请求参数和响应内容的正确编码...

    Unity 字符串与Url 转换扩展

    "xxx".Convert2StrFromUrlStr 把Url编码格式的字符串转为普通字符串:"%E6%B5%8B%E8%AF%95%E5%AD%97%E7%AC%A6%E4%B8%B2%2B-%E2%80%94%E2%80%94_%40%23%24%25" 就会被解码为:"测试字符串+-——_@#$%

    PB 进制转换 url编码 urlencode urldecode 数组排序

    自己写的,可能有bug,请大家一块学习 环境为PB12 函数(及参数) 作用 arraysort 对一维数组进行排序 decto 将十进制数字转成其它进制字符串 ...urlencode 将指定字符串以进行指定字符集url编码 涨价了,哈哈

    LoadRunner中转换字符串到URL编码

    - 在处理URL编码时,需要注意编码的范围,通常非字母数字字符和空格都需要编码。 - 由于LoadRunner的脚本是运行在服务器上的,所以调用外部程序可能涉及安全性问题,应确保工具的安全性和兼容性。 - 使用自定义...

    ios 对url中的特殊字符进行编码和反编码

    ios 对url中的特殊字符进行编码和反编码

    易语言URL编解码

    1. 字符编码:URL编解码通常基于ASCII编码,但现代的URL标准(如RFC3986)推荐使用UTF-8编码。在易语言中,你需要选择合适的字符编码方式来处理非ASCII字符。 2. 转义字符:易语言需要识别哪些字符需要转义,并用...

    unicode格式的字符串进行URL编码

    但是找遍了精易模块和百度都没有说有这这个命令 让Unicode字符串(易语言里面以字节集表示)直接进行正确的URL编码。所以就自己写了一个JavaScript脚本 进行URL编码。原理就是先把unicode字符串转换成BASE64编码文本。...

    解决字符编码的过滤器

    3. **易于维护**:集中管理字符编码设置,便于后期维护和升级。 #### 五、示例代码解析 下面是一个具体的示例代码,展示了如何在Struts2应用中配置一个名为`encodingfilter`的过滤器,并指定其作用于所有请求路径...

    URL编码 URL编码

    在实际编程中,大多数编程语言都提供了内置函数来自动进行URL编码和解码,如JavaScript的`encodeURIComponent()`和`decodeURIComponent()`,Python的`urllib.parse.quote()`和`urllib.parse.unquote()`等。...

    字符编码过滤器

    它确保了请求和响应数据在处理过程中使用一致的字符编码,从而避免乱码问题。本篇文章将详细探讨Java过滤器(Filter)的概念,字符编码的重要性,以及如何通过封装工具类来实现统一的字符编码。 首先,让我们理解...

    二进制,十六进制,Base64, URL编码,字符编码,反查工具

    下面将详细解释标题和描述中提到的几个关键概念:二进制、十六进制、Base64、URL编码以及字符编码,并结合反查工具的使用进行讨论。 首先,二进制是一种最基础的数字系统,它只包含两个数字:0和1。计算机内部的...

    URL中如果含有中文等非ASCII字符

    由于URL需要在网络上传输,为了确保传输过程中的兼容性和正确性,任何非ASCII字符都需要被编码为特定的形式,这个过程称为URL编码(URLEncode)。URL编码将非安全或不可见的字符转换成百分号“%”后跟两个十六进制...

    JAVA字符编码系列三[借鉴].pdf

    Java开发者在处理字符编码时,需要理解各种编码之间的关系和转换规则,特别是在跨平台和国际化的应用中,正确处理字符编码至关重要,可以避免乱码问题,确保数据的准确传输和存储。同时,了解这些知识也有助于解决...

    全能字符编码转换工具

    在信息技术领域,字符编码是至关重要的组成部分,它决定了计算机如何存储和显示文本。这款工具支持多种编码格式,包括GBK、GB2312、Unicode、UTF-8以及UCS和ANSI等,覆盖了国内外常见的字符集标准。 GBK编码,全称...

    字符集编码查询/反查工具

    在字符编码查询中,通常会将字符转换为二进制或十六进制进行展示和比较,因为它们更紧凑,更容易进行计算。 Base64是一种用于将二进制数据编码为ASCII字符串的方法,常用于在电子邮件等文本环境中传输非ASCII字符...

    C++URL编码和解码

    总的来说,C++中处理URL编码和解码需要理解HTTP协议和相关标准,掌握基本的字符串操作技巧,以及可能涉及的字符集问题。通过自定义实现或利用现有库,我们可以方便地在C++程序中实现URL的编码和解码功能。

    解析URL和文件的编码方式

    此外,Java的`java.net.URLDecoder`和`java.net.URLEncoder`类也提供了URL编码和解码的功能。 文件的编码方式有时会在文件的头部或通过其他方式指定,如XML文件的`&lt;?xml version="1.0" encoding="UTF-8"?&gt;`声明。在...

    url编码gbk格式

    在处理中文字符的URL时,理解并正确使用这些工具能够避免编码和解码过程中的错误,提升程序的稳定性和用户体验。对于从事Web开发的人员来说,掌握URL编码与不同字符集的配合使用是一项必不可少的技能。

    JS实现URL编码转换中文

    URL编码(也称为百分号编码)是一种机制,用于将特殊字符(如空格、中文字符等)转换为可以在URL中安全传输的形式。在URL中,某些字符有特殊含义,不能直接使用。例如,“&”用于分隔URL参数,“=”用于表示参数名和...

Global site tag (gtag.js) - Google Analytics