[GO语言]深刻理解Golang中的String字符串

小龙in武汉 · · 2290 次点击 · · 开始浏览    
这是一个创建于 的文章,其中的信息可能已经有所发展或是发生改变。

[深刻理解Golang中的string]

说到string类型,我们往往都能很熟练地对它进行各种处理,包括迭代、随机访问和匹配等等操作。然而在工作中,我发现迭代(range会自动识别成字符)一个字符串产生的字符的类型与随机访问一个字符的类型却并不相同,为什么会这么奇怪呢?于是我决定一探究竟

string 简析

在Golang中,字符串本质上看一看做一个只读的字节切片(仅比切片少了一个Cap属性)。它的底层结构我们可以查看reflect.StringHeader得到:

type StringHeader struct {
  Data uintptr
  Len  int
}

例如针对字符串"你好",其在内存中的表示如下图所示:

nihao

Go的源文件默认使用UTF-8编码,所有的字符串字面量一般也是UTF-8编码的,故这里的编码为\xe4\xbd\xa0编码为\xe5\xa5\xbd。UTF-8编码不是我们讨论的重点,具体可参考这篇博客

这里我们运行下述代码

    s := []byte{0xe4, 0xbd, 0xa0}
    fmt.Printf("char is %s", string(s))

得到运行结果char is 你

虽然字符串并非切片,但是支持切片操作。对于同一字面量,不同的字符串变量指向相同的底层数组,这是因为字符串是只读的,为了节省内存,相同字面量的字符串通常对应于同一字符串常量。例如:

    s := "hello, world"
    s1 := "hello, world"
    s2 := "hello, world"[7:]
    fmt.Printf("%d \n", (*reflect.StringHeader)(unsafe.Pointer(&s)).Data) // 17598361
    fmt.Printf("%d \n", (*reflect.StringHeader)(unsafe.Pointer(&s1)).Data) // 17598361
    fmt.Printf("%d \n", (*reflect.StringHeader)(unsafe.Pointer(&s2)).Data) // 17598368

可以看到,三者均指向同一个底层数组。对于s1, s2由于是同一字符串常量hello, world,故指向一个底层数组,以h为起始位置;而s2是字面量hello, world的切片操作后生成的字符串,也指向同一字节底层数组,不过是以w为起始位置。

迭代字符串

当我们使用for range迭代字符串时,每次迭代Go都会用UTF-8解码出一个rune类型的字符,且索引为当前rune的起始位置(以字节为最下单位)。

    for index, char := range "你好" {
        fmt.Printf("start at %d, Unicode = %U, char = %c\n", index, char, char)
    }

得到运行结果

start at 0, Unicode = U+4F60, char = 你
start at 3, Unicode = U+597D, char = 好

随机访问字符串

当我们用下标访问字符串时,返回的值为单个字节,而我们直觉中,应该返回一个字符才合理。这还是因为string的后端数组是一个字节切片而非一个字符切片

    s := "你好"
    fmt.Printf("s[%d] = %q, hex = %x, Unicode = %U", 1, s[1], s[1], s[1])

得到运行结果

s[1] = '½', hex = bd, Unicode = U+00BD

这里我们打印出来索引位置为1的字节,为0xbd,其Unicode为U+00BD, 代表的字符为½。(你可以通过这里查询)

到底什么是rune、字符和字节

字节:即byte,它由8个位组成,即1byte = 8bit,是计算机中的基本计量单位,在Go中为byte类型,其实际上为uint8的别名

字符:字符的概念比较模糊,在Unicode中通常用code point来表示。在我的理解里,是一种信息单元(例如一个符号、字母等)

rune:其实际上是int32的别名,但是为了方便将字符与整数值区分开,从而新增了rune类型代表一个字符。

总结

  • 通过下标访问字符串时,返回的是一个字节,这往往与我们的直觉相背。所以,如果你一定要通过下标访问字符串,可以先将其转换为[]rune类型
  • 字符串可以看做是一个只读字节切片, 支持切片操作。

有疑问加站长微信联系(非本文作者)

本文来自:简书

感谢作者:小龙in武汉

查看原文:[GO语言]深刻理解Golang中的String字符串

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

2290 次点击  
加入收藏 微博
添加一条新回复 (您需要 登录 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传