【Go】空struct比较

@[toc]

引入

观察如下代码

func main() {
    a := new(struct{})
    b := new(struct{})
    println("println result: ", a, b, a == b)

    c := new(struct{})
    d := new(struct{})
    fmt.Printf("fmt.Printf result: %p\t%p\t%t\n", c, d, c == d)
}

输出结果是

fmt.Printf result: 0xbedde0 0xbedde0    true
println result:  0xc00011df47 0xc00011df47 false

那么问题来了,为什么第一个返回false,第二个返回true,并且顺序也不一致

打印顺序

控制台可以看到两个打印颜色是不同的println的是红色,fmt.Printf是白色
因为println打印输出到os.Stderr
fmt.Printf打印输出到os.Stdout
println是Go在实现自举的时候供开发人员打印使用的,后续并不能保证其能正常工作

结果分析

看过fmt源码的话,很快意识到,可能是逃逸分析导致,我们对例子进行逃逸分析

go run -gcflags="-m -l" main.go
# command-line-arguments
.\main.go:14:10: new(struct {}) does not escape
.\main.go:15:10: new(struct {}) does not escape
.\main.go:18:10: new(struct {}) escapes to heap
.\main.go:19:10: new(struct {}) escapes to heap
.\main.go:20:12: ... argument does not escape
.\main.go:20:56: c == d escapes to heap
println result:  0xc00011df47 0xc00011df47 false
fmt.Printf result: 0x9edde0     0x9edde0        true

通过分析可得知变量a,b分配在栈中,c,d分配在堆中。
关键原因是fmt的Print方法内部涉及大量的反射相关方法的调用,会造成逃逸行为,也就是分配到堆上。

为什么逃逸后相等

这里主要和Go runtime的一个优化细节有关

// runtime/malloc.go
var zerobase uintptr

变量 zerobase 是所有 0 字节分配的基础地址。更进一步来讲,就是空(0字节)的在进行了逃逸分析后,往堆分配的都会指向 zerobase 这一个地址。
所以空 struct 在逃逸后本质上指向了 zerobase,其两者比较就是相等的,返回了 true。

为什么不逃逸不相等

这是Go团队故意设计的,不希望大家依赖这个来做判断依据

This is an intentional language choice to give implementations flexibility in how they handle pointers to zero-sized objects. If every pointer to a zero-sized object were required to be different, then each allocation of a zero-sized object would have to allocate at least one byte. If every pointer to a zero-sized object were required to be the same, it would be different to handle taking the address of a zero-sized field within a larger struct.

Pointers to distinct zero-size variables may or may not be equal.

在没逃逸的场景下,两个空 struct 的比较动作,并不是真的在比较。实际上已经在代码优化阶段被直接优化掉,转为了 false。
因此,虽然在代码上看上去是 == 在做比较,实际上结果是 a == b 时就直接转为了 false,比都不需要比了。

总结

  • 若逃逸到堆上,空结构体则默认分配的是 runtime.zerobase 变量,是专门用于分配到堆上的 0 字节基础地址。因此两个空结构体,都是 runtime.zerobase,一比较当然就是 true 了。
  • 若没有发生逃逸,也就分配到栈上。在 Go 编译器的代码优化阶段,会对其进行优化,直接返回 false。并不是传统意义上的,真的去比较了。
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 以下内容是我在学习和研究Go时,对Go的特性、重点和注意事项的提取、精练和总结,还有一些学习笔记(注:部分笔记是摘...
    科研者阅读 3,828评论 0 1
  • 目录 1.go 各种代码运行 2.go 在线编辑代码运行 3.通过 Gob 包序列化二进制数据 4.使用 ...
    杨言锡阅读 4,857评论 0 1
  • 1. 背景 packagemainimport"fmt"funcmain(){s:=[]byte("")s1:=a...
    也许会了阅读 4,898评论 0 0
  • 标签(空格分隔): 编程 Go官方文档 Using the tour 1.1 Hello, 世界 Welcome...
    uangianlap阅读 5,469评论 0 5
  • 编写和优化Go代码 本文档概述了编写高性能Go代码的最佳实践。 虽然有些讨论会提高单个服务的速度(通过缓存等),但...
    freelang阅读 6,732评论 0 4

友情链接更多精彩内容