关于go结构体的禁止比较技巧

在 Go 中,结构体是否可比较(可以使用 ==!= 进行比较)是编译器在编译期静态决定的特性。Go 的语言规范规定了如果一个结构体中包含不可比较的字段(如函数、切片、map 等),那么这个结构体整体也不可比较。

回到问题:为什么 _ [0]func() 能禁止结构体比较?

type Pointer struct {
	_ [0]func() // nocmp

	v unsafe.Pointer
}
分析如下:
  1. [0]func() 是一个长度为 0 的数组,元素类型是函数类型 func()

    • 函数类型在 Go 中是不可比较的。
      var f1 func() = func() {}
      var f2 func() = func() {}
      fmt.Println(f1 == f2) // 编译错误!
      
  2. 即使该数组长度为 0,只要其元素类型是不可比较的,整个结构体就不可比较

    • 即使这个数组字段实际上不占用任何内存空间,也不会影响运行时行为,但它会使得整个 struct 不可比较。
  3. 匿名字段 _表示忽略名

    • 使用_ 忽略字段名意味着你不能访问它,也不会与其它字段冲突,仅用于控制结构体的属性。

总结:为什么这么做有效?

原因点描述
Go 结构体可比较性由字段决定只要有一个字段不可比较,整个结构体就不可以比较。
函数类型本身不可比较[0]func() 中的 func() 类型是不可比较的。
空数组不影响内存布局[0]func() 不会影响结构体内存大小和对齐。
下划线 _ 避免命名污染字段不可被访问,只用于影响类型属性。

实验验证

以下代码将无法通过编译:

type S struct {
    _ [0]func()
}

func main() {
    var a, b S
    fmt.Println(a == b) // 编译错误:invalid operation
}

输出:

invalid operation: a == b (struct containing [0]func() cannot be compared)

应用场景

这种技巧常用于需要禁止用户误用结构体进行比较的场景,尤其是在封装了一些内部状态或同步逻辑的情况下。例如:

  • 同步原语(如 Mutex、AtomicPointer)
  • 包含指针或闭包的结构体
  • 强调“引用语义”而非“值语义”的类型

结论_ [0]func() 是一种经典的 Go 技巧,用于让结构体不可比较,从而避免开发者误用 ==!= 比较结构体实例。


额外多少一句, _ func() 也可以起到不能比较,但是实际工程中为何不使用呢

对比点_ [0]func() 更优的原因
内存布局更安全[0]T 类型字段在 Go 中被编译器视为零大小字段(zero-sized),不会影响结构体对齐或大小。
运行时行为明确func() 字段会占用一个指针大小(通常是 8 字节),影响结构体内存布局,虽然不影响功能,但可能造成误用或浪费空间。
语义清晰使用 [0]T 是一种在 Go 社区中广泛认可的“标记字段不可比较”的惯用法(idiom)。例如,在标准库中也大量使用这种方式。
避免意外赋值func() 字段可以被无意中赋值,而 [0]func() 几乎无法被误操作。

如果你看到很多开源项目或 Go 标准库中使用 _ [0]func() 来防止结构体比较,现在你知道为什么了 —— 它是一种高效、无副作用、语义清晰的最佳实践。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值