从外部包访问私有结构体字段
在 Go 中,结构体的私有字段在其所在的包之外是不可访问的定义的。但是,在某些情况下,可能需要从另一个包访问这些私有字段,例如出于测试目的。
使用反射
访问私有字段的一种方法另一个包中的字段正在使用反射。反射涉及在运行时检查对象的内部结构。以下代码片段演示了如何在 1.7 之前的 Go 版本中使用反射读取私有字段:
import ( "fmt" "reflect" ) func read_foo(f *Foo) { v := reflect.ValueOf(*f) y := v.FieldByName("y") fmt.Println(y.Interface()) }
使用 Unsafe
在 Go 1.7 及更高版本中,访问使用反射的私有字段被认为是不安全的。或者,您可以使用不安全的包来绕过访问限制。但是,强烈建议不要使用此方法,因为它不可移植,并且如果对象的结构发生更改,可能会导致意外行为。
以下代码片段展示了如何使用 unsafe 来访问和修改私有字段:
import "unsafe" func change_foo(f *Foo) { // Advance pointer by field size to reach desired member ptrTof := unsafe.Pointer(f) ptrTof = unsafe.Pointer(uintptr(ptrTof) + uintptr(8)) // 8 is the size of an int on a 64-bit machine ptrToy := (**Foo)(ptrTof) *ptrToy = nil // Nullify y }
替代方案
通常建议避免从外部包访问私有字段,因为它违反了封装原则。相反,请考虑以下替代方案:
白盒测试注意事项
如果您正在执行白盒测试,使用反射或不安全可能是可以接受的,因为您可以访问源代码。在这种情况下,您可以创建一个单独的测试包来定义访问私有字段的函数。然而,需要注意的是,这些测试并不是封闭的,如果对象的结构发生变化,它们的行为可能会出现意外。
以上是如何从外部包访问Go中的私有结构体字段?的详细内容。更多信息请关注PHP中文网其他相关文章!