reflect反射,可以通过reflect反射结构体所包含的属性和方法,然后进行一些赋值和方法的调用,灵活度比较高
package my
import (
ft "fmt"
"reflect"
)
type User struct {
Id int
Name string
}
func (user User) Print() {
ft.Println("reflect Print()")
}
func Reflect(inter interface{}) {
t := reflect.TypeOf(inter) //从接口中获取结构的类型
if k:=t.Kind();k!=reflect.Struct{//判断传入的是否是struce类型,而不是指针类型*User,指针类型报错
ft.Println("type is not true")
return
}
ft.Println("类型名称:", t.Name())
v := reflect.ValueOf(inter) //从接口中获取结构的值
for i := 0; i < t.NumField(); i++ { //遍历所包含的属性字段
f := t.Field(i) //获取到字段
val := v.Field(i).Interface()
ft.Println("字段签名:", f.Type, " 字段名称:", f.Name, " 值:", val)
}
for i := 0; i < t.NumMethod(); i++ { //遍历所绑定的方法
m := t.Method(i) //获取到方法
ft.Println("方法名称:", m.Name, " 方法签名:", m.Type)
}
}
- 结构体嵌套的反射
package my
import (
ft "fmt"
"reflect"
)
type User struct {
Id int
Name string
Info
}
type Info struct {
Age int
}
func (user User) Print(name string) {
ft.Println("reflect Print()",name)
}
func Reflect(inter interface{}) {
t := reflect.TypeOf(inter) //从接口中获取结构的类型
v := reflect.ValueOf(inter) //从接口中获取结构的值
if t.Kind() == reflect.Ptr && v.Elem().CanSet() { //传入的是指针,可以修改
v = v.Elem()
if f := v.Kind(); f == reflect.Struct {//如果字段属性是结构体
if x := v.FieldByName("Age"); x.IsValid() {
x.SetInt(888)
}
}
if f := v.FieldByName("Name"); f.Kind() == reflect.String && f.IsValid() {
f.SetString("haha")
}
if f := v.FieldByName("Id"); f.Kind() == reflect.Int && f.IsValid() {
f.SetInt(99)
}
if f := v.MethodByName("Print"); f.IsValid() {
args := []reflect.Value{reflect.ValueOf("测试")}
f.Call(args)
}
}
}
package main
import (
f "fmt"
"com.guo/mytest/something"
)
func main(){
r:=something.User{1,"haha",something.Info{100,"广州市"}}
something.Reflect(&r)
something.Reflect(r)
f.Println("修改后结果:",r)
}
有疑问加站长微信联系(非本文作者)