⤴Top⤴

Unicode

博客分类: 前端

Unicode

Unicode

ASCII

ASCII(American Standard Code for Information Interchange) 发音 /ˈæski/,是基于拉丁字母的一套电脑编码系统,标准 ASCII 至今为止使用 7 bits 共定义了 128 个字符,而扩展 ASCII 使用 8 bits 定义了 256 个字符,主要用于显示现代英语和其他西欧语言

在计算机中,所有的数据在存储和运算时都要使用二进制数表示。一个字节(byte)等于 8 个比特,即二进制位(bit),每一个二进制位有 0 和 1 两种状态,因此一个字节可以对应 256 个不同状态,而每一个状态对应一个符号,从 00000000 到 11111111。例如字母 A 对应十进制为 65,二进制则为 0100 0001。

如果要表示中文,显然一个字节是不够的,至少需要两个字节,而且还不能和 ASCII 编码冲突。所以,中国制定了 GB2312 编码,用来把中文编进去。

Unicode

Unicode 是为了解决传统的字符编码方案的局限而产生的,它为每种语言中的每个字符设定了统一并且唯一的二进制编码,以满足跨语言、跨平台进行文本转换、处理的要求。它从 0 开始,为每个符号指定一个编号,即码点(code point)

<!-- 码点 0 的符号就是 null -->
U+0000 = null

目前的 Unicode 字符分为 17 组编排,每组称为平面(Plane),而每平面拥有 65536(即 2^16) 个代码点。然而目前只用了少数平面。Unicode 只是一个字符集,它只规定了字符的二进制代码,UTF-8、UTF-16、UTF-32 都是字符编码,是具体的实现。

UTF-8

UTF-8(8-bit Unicode Transformation Format)是一种针对 Unicode 的可变长度字符编码,它可以使用 1 ~ 4 个字节表示一个符号,根据不同的符号而变化字节长度。

Unicode 编码(十六进制) UTF-8 字节流(二进制)
000000-00007F 0xxxxxxx(128 个字符与 ASCII 完全相同)
000080-0007FF 110xxxxx 10xxxxxx
000800-00FFFF 1110xxxx 10xxxxxx 10xxxxxx
010000-10FFFF 11110xxx10xxxxxx10xxxxxx10xxxxxx

UTF-8 和 UTF-16 及 UTF-32 的区别:

字符编码 是否可变长度 最小 bits
UTF-8 8
UTF-16 16
UTF-32 固定 32

UCS-2

UCS-2 编码

在 1991 年 Unicode 团队与 UCS 团队合并字符集之前,UCS 已于 1990 年公布了第一套编码方法 UCS-2,即使用 2 个字节表示已经有码点的字符。UTF-16 编码迟至 1996 年才公布,明确宣布是 UCS-2 的超集,即基本平面字符沿用 UCS-2 编码,辅助平面字符定义了 4 个字节的表示方法。而 JS 是于 1995 年诞生的,因此采用的是 UCS-2 字符编码,如果是 4 个字节的字符,会当作两个双字节的字符处理。

JS 允许采用 \uxxxx 形式表示一个字符,其中 xxxx 表示字符的 Unicode 十六进制码点。这种表示法只限于码点在 \u0000~\uFFFF 之间的字符。超出这个范围的字符,必须用两个双字节的形式表示:

// 例如汉字"𠮷"的码点是 0x20BB7,UTF-16 编码为0xD842 0xDFB7(十进制为55362 57271),需要4个字节储存

// ES5
'a' === '\u0061'; // true
'𠮷' === '\uD842\uDFB7'; // true
'𠮷' === '\u20BB7'; // false
'𠮷'.charCodeAt(0); // 55362
'𠮷'.charCodeAt(1); // 57271
'𠮷'.charCodeAt(1).toString(16); // 'dfb7' 转换为 16 进制

'𠮷'.length === 2; // true

// ES6
'𠮷' === '\u{20BB7}'; // true 可采用大括号 {} 的写法

Array.from('𠮷').length === 1; // true 可得到字符串的正确长度

codePointAt()

ES6 提供的 codePointAt() 能够正确处理 4 个字节储存的字符,返回一个字符的十进制码点:

'𠮷'.codePointAt(0); // 134071
'𠮷'.codePointAt(0).toString(16); // '20bb7'

'𠮷'.codePointAt(1); // 57271 仍然会显示后两个字节对应的十进制字符

for (let myCode of '𠮷') { // for...of循环会正确识别 32 位的 UTF-16 字符
  console.log(myCode.codePointAt(0).toString(16)); // '20bb7'
}

String.fromCodePoint()

ES5 提供了 String.fromCharCode(),但不能识别大于 0xFFFF 的码点。ES6 提供的 String.fromCodePoint() 能正确返回对应字符:

String.fromCharCode(0x20BB7); //"ஷ"

String.fromCodePoint(0x0061) === 'a': // true
String.fromCodePoint(0x20BB7) === '𠮷'; // true

at()

ES5 对字符串对象提供 charAt(),用来返回字符串指定位置的字符。该方法不能识别码点大于 0xFFFF 的字符,而目前 ES6 采用垫片库,可通过 at() 解决此问题:

'a'.charAt(0); // a
'𠮷'.charAt(0); // '�'

'a'.at(0); // a
'𠮷'.at(0); // '𠮷'

normalize()

对于附加符号,如汉语拼音的 Ǒ,可以用以下两种方法表示:

// 一个码点表示一个字符 - 如 Ǒ 的码点是 U+01D1
'Ǒ' === 'u\01D1'; // true

// 两个码点表示一个字符 - 如 Ǒ 可以写成 O(U+004F) + ˇ(U+030C)。
'Ǒ' === '\u004F\u030C'; // true

'\u01D1'==='\u004F\u030C'; // false

因此为了符合 Unicode 等价性,ES6 提供了 normalize() 转为同样的序列:

'\u01D1'.normalize() === '\u004F\u030C'.normalize(); // true

URL 编码

只有字母和数字 [0-9a-zA-Z]、一些特殊符号 “$-_.+!*’(),” (不包括双引号)、以及某些保留字,才可以不经过编码直接用于 URL。浏览器对于 URL 编码情况各异,因此针对此问题 JS 提供了三种编码方案: escape()、encodeURI() 和 encodeComponentURI()。

encodeURI()

encodeURI() 对 URL 进行完整编码,该方法不会对 ASCII 字母和数字进行编码,也不会对这些 ASCII 标点符号(: - _ . ! ~ * ‘ ( ) )进行编码,但是对于 URL 下特殊的 ASCII 标点符号(; / ? : @ & = + $ , #)并不会转义,它输出符号的 UTF-8 形式,并且在每个字节前加上 %。对应解码函数为 decodeURI()。

encodeURI('佘孟都');
// "%E4%BD%98%E5%AD%9F%E9%83%BD"
encodeURI("https://tate-young.github.io/README?hello Tate")
// "https://tate-young.github.io/README?hello%20Tate"

encodeURIComponent()

encodeURIComponent() 用于对 URL 的组成部分进行个别编码,而不用于对整个 URL 进行编码。对于上述 URL 下特殊的 ASCII 标点符号也会进行编码。对应解码函数为 decodeURIComponent()。

encodeURIComponent('佘孟都')
// "%E4%BD%98%E5%AD%9F%E9%83%BD"
encodeURIComponent("https://tate-young.github.io/README?hello Tate")
// "https%3A%2F%2Ftate-young.github.io%2FREADME%3Fhello%20Tate"

Base 64

Base64 是一种最常见的二进制编码方法,用 64 个字符来表示任意二进制数据。编码原理可参考 Base64笔记

base64

base64 使用场景:

电子邮件系统一般使用 SMTP(简单邮件传输协议) 将邮件从客户端发往服务器端,邮件客户端使用 POP3(邮局协议)IMAP(因特网信息访问协议) 从服务器端获取邮件。

SMTP 协议一开始是基于纯 ASCII 文本的,对于二进制文件(比如邮件附件中的图像、声音等)的处理并不好,所以后来新增 MIME 标准来编码二进制文件,使其能够通过 SMTP 协议传输。

SMTP

通过 Base64 将图片编码写入样式内可以避免与服务器交互,避免不必要的下载和加载。其中内容以 Data URLs 形式直接写在 CSS 或嵌入到 HTML 中,如果图片较大,图片的色彩层次比较丰富,则不适合使用这种方式,因为其 Base64 编码后的字符串非常大,影响加载速度。

<!-- Data URLs 形式: url(data:文件类型;编码方式,编码后的文件内容) -->
<img alt="Embedded Image" src="..." />

字符引用

在 HTML 或 XML 文档里,如果某些 Unicode 字符在文档的当前编码方式(如ISO-8859-1)中不能直接表示,那么可以通过 字符值引用(NCR) 或者 字符实体引用 两种转义序列来表示这些不能直接编码的字符。

字符值引用 NCR

字符值引用(Numeric Character Reference) 组成结构为 &# + Unicode 码点 + ;:

<!-- 「中国」二字分别是 Unicode 字符 U+4E2D 和 U+56FD,十六进制表示的码点数值「4E2D」和「56FD」就是十进制的「20013」和「22269」 -->
&#x4e2d;&#x56fd;
&#20013;&#22269;

NCR 可通过 replace() 替换的方式进行解码:

var regex_num_set = /&#(\d+);/g;
var str = "Tate: &#20312;&#23391;&#37117;"

str = str.replace(regex_num_set, function(match, p1) {
  return String.fromCharCode(p1)
})
// str --> "Tate: 佘孟都"

字符实体引用 CER

字符实体引用(Character Entity Reference) 组成结构为 &# + name + ;

名字 字符值引用 字符 十进制编码 含义
quot &#34; x22 (34) 双引号
amp &#38; & x26 (38) &
apos &#39; x27 (39) 撇号
lt &#60; < x3C (60) 小于号
gt &#62; > x3E (62) 大于号

Emoji 表情

使用方式汇总

Emoji 在上个世纪九十年代,由日本电信商引入服务,最早用于在短消息之中插入表情。2007 年,苹果公司的 iPhone 支持了 Emoji,导致它在全世界范围的流行。早期的 Emoji 是将一些特定的符号组合替换成图片,比如将 :) 替换成 😀。这种方法很难标准化,能够表达的范围也有限。2010 年,Unicode 开始为 Emoji 分配码点。也就是说,现在的 Emoji 符号就是一个文字,它会被渲染为图形。

Unicode 只是规定了 Emoji 的码点和含义,并没有规定它的样式。举例来说,码点 U+1F600 表示一张微笑的脸,但是这张脸长什么样,则由各个系统自己实现,所以一定要注意兼容性

😀 😃 😄 😁 😆 😅 😂 🤣 😊 😇 … 查看各种表情在不同系统的显示情况和对应的 Unicode,可以前往 Full Emoji List 👈

Emoji 的具体使用方式可以分为以下几点:

  1. 复制粘贴 - 从其他 Emoji 汇总网站上复制过来,粘贴到需要使用的地方即可。如 getemoji.com
  2. 直接输入字元 - 类似 👋 🤚 🖐 ✋
  3. 码点输入 - 以 HTML 网页为例,将码点 U+1F600 写成 HTML NCR 的形式 &#128512;(十进制)或 &#x1F600;(十六进制)。在线转换地址为 ifreesite 或者 Amp What
  4. JavaScript 输入 - 可以使用 node-emoji 库实现,如 emoji.get('coffee')
  5. CSS 插入 - 引入样式文件即可,如下
<!-- CSS 插入 Emoji -->
<link href="https://afeld.github.io/emoji-css/emoji.css" rel="stylesheet">
<i class="em em-baby"></i>

组合 - U+200D 零宽连字

零宽连字(Zero-width joiner) 是一个控制字符,放在某些需要复杂排版语言(如阿拉伯语、印地语)的两个字符之间,使得这两个本不会发生连字的字符产生了连字效果,零宽连字符的 Unicode 码位是 U+200D (HTML: &#8205;,&zwj;):

const man = '👨'
const woman = '👩'
const child = '👧'
// 注意要转换为 js 支持的 \uxxxx 形式
const zwj = '\u200D'

console.log(man + zwj + woman + zwj + child) // 👨‍👩‍👧 温馨的三口之家
// 不支持的系统则会顺序显示这三个绘文字(👨👩👧)

😂前綴 “\u”(\u1F602)更改為 “U+”(U+1F602)即可完成 Emoji 与 Unicode 的转换,反之亦然

肤色 - 菲茨帕特里克修饰符

Unicode 8.0 中加入了 5 个修饰符,用来调节人形表情的肤色。这些叫做绘文字菲茨帕特里克修饰符(Emoji Modifier Fitzpatrick):

字元名称 码点(code point) 字元 UTF-16 (使用 JavaScript 字串) UTF-8
FITZ-1-2 U+1F3FB 🏻 0xD83C 0xDFFB (‘\uD83C\uDFFB’) 0xF0 0x9F 0x8F 0xBB
FITZ-3 U+1F3FC 🏼 0xD83C 0xDFFC (‘\uD83C\uDFFC’) 0xF0 0x9F 0x8F 0xBC
FITZ-4 U+1F3FD 🏽 0xD83C 0xDFFD (‘\uD83C\uDFFD’) 0xF0 0x9F 0x8F 0xBD
FITZ-5 U+1F3FE 🏾 0xD83C 0xDFFE (‘\uD83C\uDFFE’) 0xF0 0x9F 0x8F 0xBE
FITZ-6 U+1F3FF 🟿(黑) 0xD83C 0xDFFF (‘\uD83C\uDFFF’) 0xF0 0x9F 0x8F 0xBF

以下是不同的 Emoji 在加上菲茨帕特里克修饰符之后的比较表:

码点 默认 菲茨-1-2 菲茨-3 菲茨-4 菲茨-5 菲茨-6
U+1F466: 男孩 👦 👦🏻 👦🏼 👦🏽 👦🏾 👦🏿
U+1F467: 女孩 👧 👧🏻 👧🏼 👧🏽 👧🏾 👧🏿
U+1F468: 男人 👨 👨🏻 👨🏼 👨🏽 👨🏾 👨🏿
U+1F469: 女人 👩 👩🏻 👩🏼 👩🏽 👩🏾 👩🏿

由于菲茨 1 和 2 类型颜色太相近,不易区分,索性合并到一起

const adult = '🧑'
const fitz6 = '\uD83C\uDFFF'
const black = adult + fitz6 // 直接加上对应的菲茨类型

console.log(black) // 🧑🏿 surprise

不是所有 Emoji 都能加上肤色,只有跟「人类」有关的 Emoji 才能这样用,例如 Emoji 中有出现脸、手、脚、身体的,都可以这样用

React 动态显示 Emoji 的问题

React 使用 Emoji 遇到的问题,场景是需要通过变量来显示 Emoji,如下:

const emoji = '&#128512;'

function ReactComponent(props) {
  return (
    <div>
      {/* 正常显示 😀,直接采用 NCR */}
      <h1>Copy & paste: &#128512;</h1>
      {/* 错误显示,会直接显示为字符串 &#128512; */}
      <span>{emoji}</span>
      {/* 正常显示 😀,通过 dangerouslySetInnerHTML */}
      <span dangerouslySetInnerHTML={\{ __html: emoji }\} />
    </div>
  )
}

Amp What is a quick, interactive reference of 33,212 HTML character entities and common Unicode characters

参考链接

  1. 字符编码笔记:ASCII,Unicode 和 UTF-8 By 阮一峰
  2. 关于 URL 编码 By 阮一峰
  3. Unicode 与 JavaScript 详解 By 阮一峰
  4. ES6 - 字符串的扩展 By 阮一峰
  5. Base64 笔记 By 阮一峰
  6. Base64 编码原理与应用 By youngsterxyf
  7. stackoverflow - What’s the difference between ASCII and Unicode?
  8. Our Code World - Encode and Decode HTML entities using pure Javascript