在golang中定义结构体时,如果字段名的首字母大写,那么这个字段就是可导出的,即可以在包外访问并修改的,而如果字段名的首字母小写,这个字段就是不可导出的。但其实,通过golang提供的反射,我们是可以在包外访问并操作到结构体中的非导出字段的。
首先在包aboutGoTest下定义一个结构体Student,包含非导出字段name和导出字段Age,然后初始化一个外部包可访问的Student对象:
package aboutGoTest
type Student struct {
name string
Age int
}
var TestStudent = Student{
name: "jack",
Age: 12,
}
在另一个包aboutGo中尝试访问TestStudent的各个字段:
package main
import (
at "aboutGoTest"
"fmt"
)
func main() {
fmt.Printf("testStudent.name: %v\n", at.TestStudent.name)
}
报错:
# command-line-arguments
../unexportedField.go:10:53: aboutGoTest.TestStudent.name undefined (cannot refer to unexported field or method name)
因此常规访问方式是访问不了name字段的,更不要说修改了。
取地址访问/修改
代码:
package main
import (
at "aboutGoTest"
"fmt"
"reflect"
)
func main() {
fmt.Printf("testStudent.name: %v\n", getUnExportedField(&at.TestStudent, "name"))
}
func getUnExportedField(ptr interface{
}, fieldName string) reflect.Value {
v := reflect.ValueOf(ptr).Elem().FieldByName(fieldName)
return v
}
结果:
testStudent.name: jack
可见,通过反射机制可以成功取出name字段。
这里,通过reflect.Value.Elem.FieldByName获取的v是非导出字段反射对象。
注意,reflect.Val