登录
首页 >  Golang >  Go问答

能通过fmt.Println函数来改变数组的值吗?

来源:stackoverflow

时间:2024-03-19 16:03:26 428浏览 收藏

在 Go 语言中,使用 fmt.Println 函数打印数组值时,如果该数组通过 unsafe 指针强制转换为不匹配的类型,则可能会导致意外的结果。这是因为底层数组的内存布局与预期不符,导致从内存中读取的字节被错误解释。因此,使用 unsafe 时需要谨慎,确保类型转换正确,否则可能会产生不可预料的行为。

问题内容

var x struct {
    a bool
    b int16
    c []int32
}

func main() {
    //a := []int32{1 << 9}
    //x.c = a
    pb := (*[]int8)(unsafe.Pointer(uintptr(unsafe.Pointer(&x)) + unsafe.Offsetof(x.c)))
    *pb = []int8{5}
    println(x.c[0])  // 5
    println(x.c[0])  // 5
    fmt.Println(x.c[0])  // 5
    fmt.Println(x.c[0])  // 327685 why????????
}

fmt.println 相同的 x.c[0] 两次,但得到不同的结果。为什么?


解决方案


使用 unsafe,您放置了一个 int8 切片,其底层数组在堆栈中分配,以代替 int32 切片。因此,现在您遇到的情况是,运行时认为存在一个 int32 值数组,但实际上该数组是一个 int8 值数组。因此,当 fmt.Println 从中读取时,您会读取包含您放置在那里的 5 的字节,以及恰好在那里的任何垃圾。

使用不同的值运行几次,并检查位模式。您将得到一个 LSB 始终为 5 的值。您正在从放置 int8 值 5 的内存位置读取 int32。剩余的三个字节是堆栈剩余值。

如果您使用 []int8{5,0,0,0} 初始化切片,您将始终得到 5。

终于介绍完啦!小伙伴们,这篇关于《能通过fmt.Println函数来改变数组的值吗?》的介绍应该让你收获多多了吧!欢迎大家收藏或分享给更多需要学习的朋友吧~golang学习网公众号也会发布Golang相关知识,快来关注吧!

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