在 Go 中访问私有字段:友元包和白盒测试的解决方案
在 Go 中,结构体字段通常默认是私有的,保护它们不被外部包访问。但是,在某些情况下,出于白盒测试或其他正当原因,您可能需要从另一个包访问私有字段。
方法 1:使用反射(Go 1.7 之前的版本)
在 Go 1.7 之前,reflect 包提供了一种使用反射访问和修改未导出字段的方法。下面的代码演示了如何使用reflect修改私有字段:
func read_foo(f *Foo) { v := reflect.ValueOf(*f) y := v.FieldByName("y") fmt.Println(y.Interface()) }
但是,使用反射修改未导出的字段会在Go 1.7及更高版本中触发panic。
方法2 :指针操作(Go 版本 1.7 及更高版本)
Go 中1.7 及更高版本,reflect 包已更新为禁止修改其他包中未导出的字段。然而,有一种危险的方法涉及使用 unsafe 包进行指针操作:
func change_foo(f *Foo) { ptrTof := unsafe.Pointer(f) ptrTof = unsafe.Pointer(uintptr(ptrTof) + uintptr(8)) ptrToy := (**Foo)(ptrTof) *ptrToy = nil }
警告: 强烈建议不要使用 unsafe 来实现此目的。它是不可移植的,可能会因数据结构的更改而中断,并且可能会干扰垃圾收集。
白盒测试
对于白盒测试,其中如果您需要访问包中的私有字段,可以使用 _test.go 命名约定。任何以 _test.go 结尾的文件仅在运行测试时才会被编译。这允许您访问私有字段进行白盒测试,同时保持它们对于生产代码私有。
// mypackage/foo_test.go package mypackage import ( "testing" ) func TestFoo(t *testing.T) { f := new(Foo) // Access and modify private fields within Foo }
结论
从另一个包访问私有字段应该是谨慎行事且仅出于正当理由。虽然反射或不安全的指针操作等方法可能提供解决方法,但它们具有显着的风险和缺点。对于白盒测试,_test.go 命名约定提供了一种更安全、更方便的方法来访问包内的私有字段。
以上是如何出于测试和其他合法目的访问 Go 中的私有字段?的详细内容。更多信息请关注PHP中文网其他相关文章!