登录
首页 >  Golang >  Go问答

golang是如何实现字节和字符串之间的转换的?

来源:stackoverflow

时间:2024-04-11 08:33:33 201浏览 收藏

一分耕耘,一分收获!既然都打开这篇《golang是如何实现字节和字符串之间的转换的?》,就坚持看下去,学下去吧!本文主要会给大家讲到等等知识点,如果大家对本文有好的建议或者看到有不足之处,非常欢迎大家积极提出!在后续文章我会继续更新Golang相关的内容,希望对大家都有所帮助!

问题内容

我无法通过检查生成的程序集得到答案:

{
        a := []byte{'a'}
        s1 := string(a)
        a[0] = 'b'
        fmt.Println(s1) // a
    }

    {
        a := "a"
        b := []byte(a)
        b[0] = 'b'
        fmt.Println(a) // a
    }

为什么会发生观察到的行为? 有没有关于 go 如何解释这些代码行的描述? go编译器对类型转换做了什么?


解决方案


这与其说是编译器问题,不如说是语言规范问题。编译器有时会做一些奇怪的事情——这里重要的是,无论编译器最终吐出什么机器代码,它都遵循语言规范中规定的规则。

正如评论中提到的,language specification 定义了 byte 切片与 string 类型之间的转换,如下所示:

将字节切片转换为字符串类型会生成一个字符串,该字符串的连续字节是该切片的元素。

将字符串类型的值转换为字节类型的切片会生成一个切片,该切片的连续元素是字符串的字节。

为了理解示例的行为,您还必须阅读 string 类型的定义,also in the specification

字符串是不可变的:一旦创建,就不可能更改字符串的内容。

因为 []byte 是可变的,所以在与 string 进行转换时,幕后 go 必须复制相关数据。这可以通过打印 []byte 对象的第 0 个元素的地址以及指向 string 对象中数据的第一个元素的指针来验证。这是一个示例(和 Go Playground version):

package main

import (
    "fmt"
    "reflect"
    "unsafe"
)

func main() {
    a := "a"
    b := []byte(a)
    ah := (*reflect.stringheader)(unsafe.pointer(&a))
    fmt.printf("a: %4s @ %#x\n", a, ah.data)
    fmt.printf("b: %v @ %p\n\n", b, b)

    c := []byte{'a'}
    d := string(c)
    dh := (*reflect.stringheader)(unsafe.pointer(&d))
    fmt.printf("c: %v @ %p\n", c, c)
    fmt.printf("d: %4s @ %#x\n", d, dh.data)
}

输出如下所示:

a:    a @ 0x4c1ab2
b: [97] @ 0xc00002c008

c: [97] @ 0xc00002c060
d:    a @ 0x554e21

请注意,string[]byte 的指针位置不同且不重叠。因此,对 []byte 值的更改不会以任何方式影响 string 值。

好吧,技术上结果不一定是这样,因为我没有在示例中对 bc 的值进行任何更改。 技术上编译器可以采取捷径,简单地调用 b ,长度 = 1 []bytea 开始于相同的内存地址。但如果我这样做的话,这种优化就不会被允许:

package main

import (
    "fmt"
    "reflect"
    "unsafe"
)

func main() {
    a := "a"
    b := []byte(a)
    b[0] = 'b'
    ah := (*reflect.stringheader)(unsafe.pointer(&a))
    fmt.printf("a: %4s @ %#x\n", a, ah.data)
    fmt.printf("b: %v @ %p\n\n", b, b)
}

输出:

a:    a @ 0x4c1ab2
b: [98] @ 0xc00002c008

请拨打 Go Playground 查看实际操作。

今天关于《golang是如何实现字节和字符串之间的转换的?》的内容介绍就到此结束,如果有什么疑问或者建议,可以在golang学习网公众号下多多回复交流;文中若有不正之处,也希望回复留言以告知!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>