反射reflection
- 反射可大大提高程序的灵活性,使得interface{}有更大的发挥余地
- 反射使用TypeOf和ValueOf函数从接口中获取目标对象信息
- 反射会将匿名字段作为独立字段(匿名字段本质)
- 想要利用反射修改对象状态,前提是interface.data是settable,即pointer-interface
- 通过反射可以动态调用方法
package main
import (
"fmt"
"reflect"
)
type User struct {
Id int
Name string
Age int
}
func (u User) Hello() {
fmt.Println("Hello world.")
}
func main() {
u := User{1, "OK", 12}
Info(u)
}
func Info(o interface{}) {
t := reflect.TypeOf(o)
fmt.Println("Type:", t.Name())
//获取字段的名称信息
v := reflect.ValueOf(o)
fmt.Println("Fields:")
for i := 0; i < t.NumField(); i++ {
f := t.Field(i)
val := v.Field(i).Interface()
fmt.Printf("%6s: %v = %v\n", f.Name, f.Type, val)
}
//获取字段的类型信息
for i := 0; i < t.NumMethod(); i++ {
m := t.Method(i)
fmt.Printf("%6s: %v\n", m.Name, m.Type)
}
//获取字段的方法信息
}
运行结果如下:
Type: User
Fields:
Id: int = 1
Name: string = OK
Age: int = 12
Hello: func(main.User)
package main
import (
"fmt"
"reflect"
)
type User struct {
Id int
Name string
Age int
}
type Manager struct {
User
title string
}
func main() {
m := Manager{User: User{1, "OK", 12}, title: "123"}
t := reflect.TypeOf(m)
fmt.Printf("%#v\n", t.Field(0)) //取出第一个,也就是0号
fmt.Printf("%#v\n", t.Field(1)) //取出第二个,也就是1号
//通过看Anonymous的值来判断是否为匿名字段,true为匿名字段,false为非匿名字段
/*还有一个更高级的方法*/
fmt.Printf("%#v\n", t.FieldByIndex([]int{0, 0}))
//这样通过二维数组的下标(0,0)可以取得第1个位置的第1个字段,即User的Id字段
}
运行结果如下:
reflect.StructField{Name:“User”, PkgPath:"", Type:(*reflect.rtype)(0x4a27a0), Tag:"", Offset:0x0, Index:[]int{0}, Anonymous:true}
reflect.StructField{Name:“title”, PkgPath:“main”, Type:(*reflect.rtype)(0x497240), Tag:"", Offset:0x20, Index:[]int{1}, Anonymous:false}
reflect.StructField{Name:“Id”, PkgPath:"", Type:(*reflect.rtype)(0x496d40), Tag:"", Offset:0x0, Index:[]int{0}, Anonymous:false}