func main() {
s := "Hi小智加油!"
fmt.Println("len(s):",len(s)) //len(s): 15 为什么是15呢?
for _, v := range []byte(s) {
fmt.Printf("%X ",v) //%X 转为16进制
//48 69 E5 B0 8F E6 99 BA E5 8A A0 E6 B2 B9 21
//猜测从第三个开始(E5 B0 8F) -> 小 ,后面都是三个字节代表一个汉字
//其实utf-8 是使用的可变长度来编码的(世界上这么多语言,如果全部都是3个直接来编码,太浪费空间)
}
}
通过打印我们猜测,这个byte数组中,中文汉字是使用3个字节来编码
验证:上述猜想
func main() {
for k, v := range s { //v is a rune
fmt.Printf("(%d,%X)",k,v)
//(0,48)(1,69)(2,5C0F)(5,667A)(8,52A0)(11,6CB9)(14,21)
}
}
如图for k, v := range s
的v
是一个int32类型:
发现从第三个开始汉字,下标正确2,但是下一个就变成了5,这说明,上面转为byte数组时我们的猜想是正确的,
一个汉字使用了3个byte来表示,但是这样遍历 根本不是我们想要的,因为下标的问题.
总结:既 直接遍历字符串时,k 为 byte数组的下标 而 v 却是int32类型(rune)
//使用 utf8工具类
fmt.Println("s rune counts:",utf8.RuneCountInString(s))
//s rune counts: 7 ,发现是s的长度是我们想要的了
既然可以得到正确的长度了,尝试一个字一个字解码
bytes := []byte(s)
for len(bytes)>0 {
ch, size := utf8.DecodeRune(bytes)
//解码byte数组 返回rune和size长度
bytes = bytes[size:]
//截取byte
fmt.Printf("%c " ,ch)
}
H i 小 智 加 油 ! //发现每个字符都被解码出来了
在go中,rune就相当于Java中的char,而rune是uft8编码
正常遍历一个string的方法
fmt.Println();
for k, v := range []rune(s) { //直接将string转为rune切片
fmt.Printf("(%d %c)",k,v) //%c 可以将rune转为 字符串
}
(0 H)(1 i)(2 小)(3 智)(4 加)(5 油)(6 !) 得到了我们想要的答案.
这里直接跳过了这个test的解题思路,如果没有看过的同学, 点击上篇查看
func findMaxNoRepeatString(s string) int {
start := 0
keysIndex := make(map[rune]int)
lenth := 0
for i, v := range []rune(s) { //将byte换成rune即可
lastIndex, ok := keysIndex[v]
if ok && lastIndex >= start {
start = lastIndex + 1
} else {
lenth = i - start + 1
}
keysIndex[v] = i
}
return lenth
}
作者所有的学习源码在 go学习源码github地址,如果觉得有用的话帮小智贡献一个star?