嵌入指向结构体的指针与嵌入结构体本身
在给定的场景中,结构体类型 A 用作结构体类型指针并且仅包含指针接收器,将结构类型 B 嵌入为 B 与 *B 之间存在区别。
零值行为
最明显的区别在于所得结构的零值。当按原样嵌入 B 时,类型 A 的零值将包含类型 B 的嵌入对象,该对象也保留其零值。这使得无需事先初始化即可无缝使用嵌入字段:
type B struct { X int } func (b *B) Print() { fmt.Printf("%d\n", b.X) } type AObj struct { B } var aObj AObj aObj.Print() // prints 0
另一方面,嵌入 *B 会导致带有 nil 指针的零值,从而阻止直接字段访问:
type APtr struct { *B } var aPtr APtr aPtr.Print() // panics
字段复制
创建 A 类型的新对象时,字段复制的行为与直觉一致。嵌入 B 直接在新对象中创建嵌入对象的副本,保留独立性:
aObj2 := aObj aObj.X = 1 aObj2.Print() // prints 0, due to a separate instance of B
相反,嵌入 *B 创建指针的副本而不是底层具体对象。这意味着两个对象共享相同的 B 实例,从而影响字段修改:
aPtr.B = &B{} aPtr2 := aPtr aPtr.X = 1 aPtr2.Print() // prints 1, as both objects point to the same B
本质上,嵌入 B 创建所包含结构的不同实例,而嵌入 *B 在多个嵌入结构中共享单个具体实例。这种区别会影响零值语义和字段复制的行为。
以上是在 Go 中何时嵌入指向结构体的指针与结构体本身?的详细内容。更多信息请关注PHP中文网其他相关文章!