// Copyright 2009 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.// go/src/unicode/utf8/utf8.go
// version 1.7// 关于 UTF-8 编码方式请参考:http://www.cnblogs.com/golove/p/3222096.htmlpackage utf8// 编码所需的基本数字
const (RuneError = '\uFFFD'     // 错误的 Rune 或 Unicode 代理字符RuneSelf  = 0x80         // ASCII 字符范围MaxRune   = '\U0010FFFF' // Unicode 码点的最大值UTFMax    = 4            // 一个字符编码的最大长度
)// Unicode 代理字符对 UTF-8 编码而言是无效的。
const (surrogateMin = 0xD800surrogateMax = 0xDFFF
)// 用词说明:
// 单字节字符:该字符的 UTF-8 编码需要一个字节存放
// 双字节字符:该字符的 UTF-8 编码需要两个字节存放
// 三字节字符:该字符的 UTF-8 编码需要三个字节存放
// 四字节字符:该字符的 UTF-8 编码需要四个字节存放
// 字符首字节:某字符的 UTF-8 编码中的第一个字节
// 字符次字节:某字符的 UTF-8 编码中的第二个字节
// 字符后续字节:某字符的 UTF-8 编码中首字节后面的其它字节const (// 位标记(用于判断字节有效性)t1 = 0x00 // 0000 0000 单字节字符的首字节标记(二进制以 0     开头)tx = 0x80 // 1000 0000 所有字符的后续字节标记(二进制以 10    开头)t2 = 0xC0 // 1100 0000 双字节字符的首字节标记(二进制以 110   开头)t3 = 0xE0 // 1110 0000 三字节字符的首字节标记(二进制以 1110  开头)t4 = 0xF0 // 1111 0000 四字节字符的首字节标记(二进制以 11110 开头)t5 = 0xF8 // 1111 1000 好像未使用// 位掩码(用于获取标记之外的二进制位)maskx = 0x3F // 0011 1111 所有字符的后续字节掩码mask2 = 0x1F // 0001 1111 双字节字符的首字节掩码mask3 = 0x0F // 0000 1111 三字节字符的首字节掩码mask4 = 0x07 // 0000 0111 四字节字符的首字节掩码rune1Max = 1<<7 - 1  // 单字节字符的总数(127   个)rune2Max = 1<<11 - 1 // 双字节字符的总数(2047  个)rune3Max = 1<<16 - 1 // 三字节字符的总数(65535 个)// UTF-8 字符的后续字节的一般取值范围locb = 0x80 // 1000 0000hicb = 0xBF // 1011 1111// 字符首字节分类标记,用于将所有的字符首字节分成下面九类,分别处理。// 以下十六进制常量的高位和低位分别表示不同的含义:// 高位:“次字节取值范围列表”的索引,如果高位是 F 则表示字符是单字节字符// 低位:字符的编码长度,如果高位是 F 则低位表示单字节字符的状态:有效、无效xx = 0xF1 // 无索引,长度 1,对应无效 UTF-8 编码as = 0xF0 // 无索引,长度 1,对应普通 ASCII 字符s1 = 0x02 // 索引 0, 长度 2,对应普通“双字节字符”的首字节s2 = 0x13 // 索引 1, 长度 3,对应特殊“双字节字符”的首字节 0xE0(用于编码长度跨越)s3 = 0x03 // 索引 0, 长度 3,对应普通“三字节字符”的首字节s4 = 0x23 // 索引 2, 长度 3,对应特殊“三字节字符”的首字节 0xED(用于代理区检测)s5 = 0x34 // 索引 3, 长度 4,对应特殊“四字节字符”的首字节 0xF0(用于编码长度跨越)s6 = 0x04 // 索引 0, 长度 4,对应普通“四字节字符”的首字节s7 = 0x44 // 索引 4, 长度 4,对应特殊“四字节字符”的首字节 0xF4(用于范围检测)
)// first 是关于 UTF-8 字符中首字节的编码信息。
// 将所有的首字节进行分类,分为:xx、as、s1、s2、s3、s4、s5、s6、s7 九类,
// 其中 xx 代表无效首字节,s1 代表双字节字符的首字节
// s2、s3、s4 代表三字节字符的首字节
// s5、s6、a7 代表四字节字符的首字节
var first = [256]uint8{//   1   2   3   4   5   6   7   8   9   A   B   C   D   E   Fas, as, as, as, as, as, as, as, as, as, as, as, as, as, as, as, // 0x00-0x0Fas, as, as, as, as, as, as, as, as, as, as, as, as, as, as, as, // 0x10-0x1Fas, as, as, as, as, as, as, as, as, as, as, as, as, as, as, as, // 0x20-0x2Fas, as, as, as, as, as, as, as, as, as, as, as, as, as, as, as, // 0x30-0x3Fas, as, as, as, as, as, as, as, as, as, as, as, as, as, as, as, // 0x40-0x4Fas, as, as, as, as, as, as, as, as, as, as, as, as, as, as, as, // 0x50-0x5Fas, as, as, as, as, as, as, as, as, as, as, as, as, as, as, as, // 0x60-0x6Fas, as, as, as, as, as, as, as, as, as, as, as, as, as, as, as, // 0x70-0x7F//   1   2   3   4   5   6   7   8   9   A   B   C   D   E   Fxx, xx, xx, xx, xx, xx, xx, xx, xx, xx, xx, xx, xx, xx, xx, xx, // 0x80-0x8Fxx, xx, xx, xx, xx, xx, xx, xx, xx, xx, xx, xx, xx, xx, xx, xx, // 0x90-0x9Fxx, xx, xx, xx, xx, xx, xx, xx, xx, xx, xx, xx, xx, xx, xx, xx, // 0xA0-0xAFxx, xx, xx, xx, xx, xx, xx, xx, xx, xx, xx, xx, xx, xx, xx, xx, // 0xB0-0xBFxx, xx, s1, s1, s1, s1, s1, s1, s1, s1, s1, s1, s1, s1, s1, s1, // 0xC0-0xCFs1, s1, s1, s1, s1, s1, s1, s1, s1, s1, s1, s1, s1, s1, s1, s1, // 0xD0-0xDFs2, s3, s3, s3, s3, s3, s3, s3, s3, s3, s3, s3, s3, s4, s3, s3, // 0xE0-0xEFs5, s6, s6, s6, s7, xx, xx, xx, xx, xx, xx, xx, xx, xx, xx, xx, // 0xF0-0xFF
}// acceptRange 给出次字节的取值范围。
type acceptRange struct {lo uint8 // 次字节最小取值hi uint8 // 次字节最大取值
}// 不同的首字节字符有不同的次字节取值范围,
// UTF-8 编码编不出这些范围之外的次字节内容。
var acceptRanges = [...]acceptRange{// 普通字符的次字节,范围之外为无效编码(即二进制位不是以 10 开头)0: {locb, hicb},// 三字节特殊字符(首字节为 0xE0)的次字节:// 如果次字节低于 0xA0 则该字符应该用两个字节表示,而不是三个字节。// 如果次字节高于 hicb 则该字节为无效编码(即二进制位不是以 10 开头)1: {0xA0, hicb},// 三字节特殊字符(首字节为 0xED)的次字节:// 如果次字节低于 locb 则该字节为无效编码(即二进制位不是以 10 开头)// 如果次字节高于 0x9F 则该字符为代理区字符([ED A0 80] - [ED BF BF])2: {locb, 0x9F},// 四字节特殊字符(首字节为 0xF0)的次字节:// 如果次字节低于 0x90 则该字符应该用三个字节表示,而不是四个字节。// 如果次字节高于 hicb 则该字节为无效编码(即二进制位不是以 10 开头)3: {0x90, hicb},// 四字节特殊字符(首字节为 0xF4)的次字节:// 如果次字节低于 locb 则该字节为无效编码(即二进制位不是以 10 开头)// 如果次字节高于 0x8F 则该字符超出 Unicode 范围(超出 MaxRune)4: {locb, 0x8F},// 相邻字符的编码长度跨越:// [11011111 10111111]          [DF BF]    // U+07FF 的后一个字符为三字节// [11100000 10100000 10000000] [E0 A0 80] // U+0800 的前一个字符为两字节// [11101111 10111111 10111111]          [EF BF BF]    // U+FFFF  的后一个字符为四字节// [11110000 10010000 10000000 10000000] [F0 90 80 80] // U+10000 的前一个字符为三字节
}// FullRune 判断 p 是否以一个完整(但不一定有效)的 UTF-8 字符开头。
// 一个无效的编码也被认为是完整字符,因为它将被转换为一个 RuneError 字符。
// 只有“编码有效但长度不够”的字符才被认为是不完整字符。
// 也就是说,只有截去一个有效字符的一个或多个后续字节,该字符才算是不完整字符。
// 举例:
// "好"     是完整字符
// "好"[1:] 是完整字符(首字节无效,转换为 RuneError 字符)
// "好"[2:] 是完整字符(首字节无效,转换为 RuneError 字符)
// "好"[:2] 是不完整字符(编码有效但长度不够)
// "好"[:1] 是不完整字符(编码有效但长度不够)
func FullRune(p []byte) bool {n := len(p)if n == 0 {return false}// 查表并计算,获取编码长度,判断 p 的长度是否满足编码长度x := first[p[0]]if n >= int(x&7) { // x&7 获取的就是编码长度// p 的长度满足编码长度,表示 p 是一个完整的字符开头。return true}// 此时 p 的长度不够,应该是不完整的字符了,但是如果 p 中是无效编码,也算完整。// 此时 n 肯定小于 4,否则长度不可能不够。// 获取首字节对应的次字节有效范围accept := acceptRanges[x>>4]if n > 1 {if c := p[1]; c < accept.lo || accept.hi < c {// 有一个无效字节,算完整字符return true} else if n > 2 && (p[2] < locb || hicb < p[2]) {// 有一个无效字节,算完整字符return true}}// 全是有效字节,但长度不够,算不完整return false
}// 功能同 FullRune,只不过参数为字符串。
func FullRuneInString(s string) bool {n := len(s)if n == 0 {return false}x := first[s[0]]if n >= int(x&7) {return true}accept := acceptRanges[x>>4]if n > 1 {if c := s[1]; c < accept.lo || accept.hi < c {return true} else if n > 2 && (s[2] < locb || hicb < s[2]) {return true}}return false
}// 解码 UTF-8 序列 p 中的第一个 Unicode 字符。
// r   :解码出的字符
// size:该字符的 UTF-8 编码长度
// 如果 p 为空,则返回 RuneError, 0
// 如果 p 为无效的 UTF-8 编码,则返回 RuneError, 1
// 无效 UTF-8 编码:UTF-8 编码不正确(比如长度不够)、结果超出 Unicode 范围、
// 编码不是最短的。
// 可以用四个字节编码一个单字节字符,但它不是最短的,比如:
// [111100000 10000000 10000000 10111000] 不是最短的,应该使用 [00111000]
func DecodeRune(p []byte) (r rune, size int) {n := len(p)if n < 1 {return RuneError, 0}// 处理单字节字符p0 := p[0]x := first[p0]if x >= as { // x 为 F0 或 F1// 生成 0x0000 或 0xFFFFmask := rune(x) << 31 >> 31// return 保留 ASCII 字符 | 保留 RuneError, 1return rune(p[0])&^mask | RuneError&mask, 1}// 处理多字节字符// 获取编码长度sz := x & 7// 获取次字节有效范围accept := acceptRanges[x>>4]// p 长度不够if n < int(sz) {return RuneError, 1}// p 长度满足// 次字节编码有效b1 := p[1]if b1 < accept.lo || accept.hi < b1 {return RuneError, 1}// 处理有效的双字节字符if sz == 2 {return rune(p0&mask2)<<6 | rune(b1&maskx), 2}// 超过双字节,第三字节编码有效b2 := p[2]if b2 < locb || hicb < b2 {return RuneError, 1}// 处理有效的三字节字符if sz == 3 {return rune(p0&mask3)<<12 | rune(b1&maskx)<<6 | rune(b2&maskx), 3}// 超过三字节,第四字节编码有效b3 := p[3]if b3 < locb || hicb < b3 {return RuneError, 1}// 处理有效的四字节字符return rune(p0&mask4)<<18 | rune(b1&maskx)<<12 | rune(b2&maskx)<<6 | rune(b3&maskx), 4
}// 功能同 DecodeRune,只不过参数为字符串
func DecodeRuneInString(s string) (r rune, size int) {n := len(s)if n < 1 {return RuneError, 0}s0 := s[0]x := first[s0]if x >= as {mask := rune(x) << 31 >> 31return rune(s[0])&^mask | RuneError&mask, 1}sz := x & 7accept := acceptRanges[x>>4]if n < int(sz) {return RuneError, 1}s1 := s[1]if s1 < accept.lo || accept.hi < s1 {return RuneError, 1}if sz == 2 {return rune(s0&mask2)<<6 | rune(s1&maskx), 2}s2 := s[2]if s2 < locb || hicb < s2 {return RuneError, 1}if sz == 3 {return rune(s0&mask3)<<12 | rune(s1&maskx)<<6 | rune(s2&maskx), 3}s3 := s[3]if s3 < locb || hicb < s3 {return RuneError, 1}return rune(s0&mask4)<<18 | rune(s1&maskx)<<12 | rune(s2&maskx)<<6 | rune(s3&maskx), 4
}// 功能同 DecodeRune,只不过解码的是最后一个字符。
func DecodeLastRune(p []byte) (r rune, size int) {end := len(p)if end == 0 {return RuneError, 0}// 处理 p 的最后一个字节start := end - 1r = rune(p[start])if r < RuneSelf { // 单字节字符直接返回return r, 1}// 一次最多遍历 4 个字节,避免因无效 UTF8 编码造成的过度循环lim := end - UTFMaxif lim < 0 {lim = 0}// 按字节反向遍历for start--; start >= lim; start-- {if RuneStart(p[start]) { // 遇到首字节编码即可break}}// 遍历完了也没遇到首字节,则解码整个 pif start < 0 {start = 0}r, size = DecodeRune(p[start:end])// 遇到无效编码,则只将最后一个字节解码为 RuneErrorif start+size != end {return RuneError, 1}// 解码成功return r, size
}// 功能同 DecodeLastRune,只不过参数为字符串
func DecodeLastRuneInString(s string) (r rune, size int) {end := len(s)if end == 0 {return RuneError, 0}start := end - 1r = rune(s[start])if r < RuneSelf {return r, 1}lim := end - UTFMaxif lim < 0 {lim = 0}for start--; start >= lim; start-- {if RuneStart(s[start]) {break}}if start < 0 {start = 0}r, size = DecodeRuneInString(s[start:end])if start+size != end {return RuneError, 1}return r, size
}// RuneLen 返回 r 的 UTF-8 编码所占用的字节数。
// 如果 r 不是一个有效的值(代理区或超出范围),则返回 -1。
func RuneLen(r rune) int {switch {case r < 0: // 超出范围return -1case r <= rune1Max: // 单字节字符范围return 1case r <= rune2Max: // 双字节字符范围return 2case surrogateMin <= r && r <= surrogateMax: // 代理区范围return -1case r <= rune3Max: // 三字节字符范围return 3case r <= MaxRune: // 四字节字符范围return 4}return -1 // 超出范围
}// EncodeRune 将 r 编码为 UTF-8 序列,结果写入 p 中(p 必须足够长,一般为 4)
// 返回写入的字节数
func EncodeRune(p []byte, r rune) int {// 负数是错误的,将其转换为无符号数,以使其超出范围,进而处理掉这个错误。switch i := uint32(r); {case i <= rune1Max: // 单字节字符p[0] = byte(r)return 1case i <= rune2Max: // 双字节字符p[0] = t2 | byte(r>>6)p[1] = tx | byte(r)&maskxreturn 2// 超出范围或代理区字符case i > MaxRune, surrogateMin <= i && i <= surrogateMax:r = RuneErrorfallthroughcase i <= rune3Max: // 三字节字符p[0] = t3 | byte(r>>12)p[1] = tx | byte(r>>6)&maskxp[2] = tx | byte(r)&maskxreturn 3default: // 四字节字符p[0] = t4 | byte(r>>18)p[1] = tx | byte(r>>12)&maskxp[2] = tx | byte(r>>6)&maskxp[3] = tx | byte(r)&maskxreturn 4}
}// RuneCount 返回 p 中的字符数(不是字节数)
// 错误的和长度无效的编码中的每一个字节都会被当做一个字符处理。
// RuneError 被视为一个字符
func RuneCount(p []byte) int {np := len(p)var n intfor i := 0; i < np; {n++c := p[i]if c < RuneSelf {i++ // 单字节字符continue}// 查表判断首字节的有效性x := first[c]if x == xx {i++ // 首字节无效,字节当做一个字符处理continue}// 首字节有效size := int(x & 7)if i+size > np {i++ // 但长度不足,字节当做一个字符处理continue}// 首字节有效,长度也够,判断后续字节的有效性accept := acceptRanges[x>>4]if c := p[i+1]; c < accept.lo || accept.hi < c { // 次字节无效size = 1} else if size == 2 { // 次字节有效,长度刚好为 2} else if c := p[i+2]; c < locb || hicb < c { // 第三字节无效size = 1} else if size == 3 { // 第三字节也有效,长度刚好为 3} else if c := p[i+3]; c < locb || hicb < c { // 第四字节无效size = 1} // 第四字节也有效,长度不是 1、2、3,肯定为 4(size == 4)i += size}return n
}// 功能同 RuneCount,只不过参数为字符串
func RuneCountInString(s string) (n int) {ns := len(s)for i := 0; i < ns; n++ {c := s[i]if c < RuneSelf {i++continue}x := first[c]if x == xx {i++continue}size := int(x & 7)if i+size > ns {i++continue}accept := acceptRanges[x>>4]if c := s[i+1]; c < accept.lo || accept.hi < c {size = 1} else if size == 2 {} else if c := s[i+2]; c < locb || hicb < c {size = 1} else if size == 3 {} else if c := s[i+3]; c < locb || hicb < c {size = 1}i += size}return n
}// RuneStart 判断 b 是否为 UTF-8 字符编码的首字节(有可能是无效字节)。
// UTF-8 编码的后续字节的二进制位都是以 10 开始的。
func RuneStart(b byte) bool { return b&0xC0 != 0x80 }// Valid 判断 p 是否完全由有效的 UTF-8 编码组成。
func Valid(p []byte) bool {// 代码同 RuneCount 类似n := len(p)for i := 0; i < n; {pi := p[i]if pi < RuneSelf {i++continue}x := first[pi]if x == xx {return false}size := int(x & 7)if i+size > n {return false}accept := acceptRanges[x>>4]if c := p[i+1]; c < accept.lo || accept.hi < c {return false} else if size == 2 {} else if c := p[i+2]; c < locb || hicb < c {return false} else if size == 3 {} else if c := p[i+3]; c < locb || hicb < c {return false}i += size}return true
}// 功能同 Valid,只不过参数为字符串
func ValidString(s string) bool {n := len(s)for i := 0; i < n; {si := s[i]if si < RuneSelf {i++continue}x := first[si]if x == xx {return false}size := int(x & 7)if i+size > n {return false}accept := acceptRanges[x>>4]if c := s[i+1]; c < accept.lo || accept.hi < c {return false} else if size == 2 {} else if c := s[i+2]; c < locb || hicb < c {return false} else if size == 3 {} else if c := s[i+3]; c < locb || hicb < c {return false}i += size}return true
}// ValidRune 判断 r 是否可以被编码成 UTF-8 序列。
// 代理区字符或超出范围则返回 false。
func ValidRune(r rune) bool {switch {case r < 0: // 超出范围return falsecase surrogateMin <= r && r <= surrogateMax: // 代理区字符return falsecase r > MaxRune: // 超出范围return false}return true
}

转载于:https://www.cnblogs.com/golove/p/5889790.html

标准库 - unicode/utf8/utf8.go 解读相关推荐

  1. GoLang之标准库unicode/utf8包

    文章目录 GoLang之标准库unicode/utf8包 1.Constants 2.DecodeRune函数 3.DecodeRuneInString函数 3.ValidRune 4.RuneLen ...

  2. C++中字符编码的转换(Unicode、UTF-8、ANSI)

    C++的项目,字符编码是一个大坑,不同平台之间的编码往往不一样,如果不同编码格式用一套字符读取格式读取就会出现乱码.因此,一般都是转化成UTF-8这种平台通用,且支持性很好的编码格式. Unicode ...

  3. 字符串和字符串标准库

    1.简介编码方式 计算机的视角,世间万物不过是一串又一串的二进制字节流,以人类认识的字符为例,每一个字符串就必须有与之对应一个二进制码,这就叫做编码.然而在发明计算机时,发明者们只考虑到计算机处理的是 ...

  4. libIconv库实现中文中字符串与GBK、Unicode、UTF-8三种编码互转

    libIconv库实现GBK.Unicode.UTF-8三种编码互转比window api更为简单,而且libIconv库跨平台. IibIconv库在windows下的编译参照windows下使用V ...

  5. Unicode 和 UTF-8关系

    unicode 就是 "与存储无关的表示",utf-8 就是 "二进制表示".一句话,utf8是对unicode字符集进行编码的一种编码方式,utf8是给uni ...

  6. unicode,ansi,utf-8,unicode big endian编码的区别

    为什么80%的码农都做不了架构师?>>>    随便说说字符集和编码 快下班时,爱问问题的小朋友Nico又问了一个问题: "sqlserver里面有char和nchar,那 ...

  7. 字符编码笔记:ASCII,Unicode和UTF-8

    很久很久以前,有一群人,他们决定用8个可以开合的晶体管来组合成不同的状态,以表示世界上的万物.他们看到8个开关状态是好的,于是他们把这称为"字节". 再后来,他们又做了一些可以处理 ...

  8. ASCII、GB2312、GBK、Unicode、UTF-8介绍和转换

    1.ASCII码 上个世纪60年代,美国制定了一套字符编码,对英语字符与二进制位之间的关系,做了统一规定.这被称为 ASCII 码,一直沿用至今.ASCII 码一共规定了128个字符的编码,比如空格S ...

  9. python与php8-详解Python中Unicode和utf-8

    在Python语言中,Uincode字符串处理一直是一个容易让人迷惑的问题.许多Python爱好者经常因为搞不清Unicode.UTF-8还有其它许许多多的编码之间的区别而大伤脑筋.本文将介绍Unic ...

最新文章

  1. 在Linux系统下如何运行notes客户端
  2. keepalive的 nopreempt 非抢占
  3. [云炬创业基础笔记]第五章创业机会评估测试6
  4. linux sed命令整行替换:将`PermitRootLogin`行替换成`PermitRootLogin yes`
  5. ActiveMQ 实现消息接收发送
  6. Android中shape的解析
  7. 什么叫云平台_为什么说云原生会成为未来企业技术变迁的趋势
  8. MATLAB中矩阵与数组的区别,点运算符的运用
  9. pandas dataframe使用query进行多个条件快速筛选
  10. MySQL 性能分析 之 联合索引(复合索引)实践分析
  11. ## CSP 201412-2 Z字形扫描(C语言)(100分)
  12. ipmi 修改服务器密码,通过ipmi方式重设dell远程控制卡的密码
  13. 全网首发:解决办法:shmget()一直失败,errno=-22
  14. PLC数据采集解决方案,BCNet数据采集
  15. 步骤安装Ubuntu 11.04用五笔
  16. 软件测试笔记本硬件,教你全面检测笔记本
  17. Porsche保时捷Taycan维修手册电路图接线图技术培训手册维修技术资料
  18. 2021-2027全球与中国电子束加速器市场现状及未来发展趋势
  19. Android 开机加速优化
  20. 以昂扬的斗志,书写青春的热血

热门文章

  1. zip分卷压缩linux命令,linux下zip分卷压缩及linux下zip分卷解压
  2. 内存 ECC 校验错误
  3. 《什么是数学》第一章自然数习题
  4. uniapp简单音频组件
  5. try-catch-finally的使用
  6. 【关于运维和网工的差别,一文说透】
  7. Microsoft.Office.Interop.Word 在某一段落前插入新的段落
  8. only regular white space (\r, \n, \t) is allowed between tokens
  9. JavaScript自定义对象
  10. HDU - 1753 - 大明A+B