目录
前言
概念:运动时动态的获取变量信息。
1、reflect
reflect.TypeOf,获取变量的类型,返回reflect.Type。
reflect.ValueOf,获取变量的值,返回reflect.Value类型。
reflect.Value.Kind,获取变量的类别,返回一个常量。
reflect.Value.Interface(),转换成interface{}类型。
示例:获取变量类型。
package main
import (
"fmt"
"reflect"
)
func Test(i interface{}) {
//反射数据类型
t := reflect.TypeOf(i)
fmt.Println("类型是", t)
//反射数据值
v := reflect.ValueOf(i)
fmt.Println("值是", v)
}
func main() {
a := "hello"
Test(a)
}
//运行结果为:
类型是 string
值是 hello
示例:
package main
import (
"fmt"
"reflect"
)
//定义结构体
type Student struct {
Name string
Age int
Score float32
}
func Test(p interface{}) {
t := reflect.TypeOf(p)
fmt.Println("类型:", t)
//类别
v := reflect.ValueOf(p)
c := v.Kind()
fmt.Println("类型:", c)
}
func main() {
var stu Student = Student{
Name: "zhangsan",
Age: 20,
Score: 80,
}
Test(stu)
fmt.Println("----------------")
var num int = 10
Test(num)
}
//运行结果为:
类型: main.Student
类型: struct
----------------
类型: int
类型: int
示例:断言处理类型转换。
package main
import (
"fmt"
"reflect"
)
type Student struct {
Name string
Age int
Score float32
}
func Test(i interface{}) {
t := reflect.TypeOf(i)
fmt.Println("类型是", t)
//类别
v := reflect.ValueOf(i)
c := v.Kind()
fmt.Println("类别是", c)
fmt.Printf("c的类型是%T\n", c)
fmt.Printf("v的类型是%T\n", v)
//转换成接口
iv := v.Interface()
fmt.Printf("iv的类型%T\n", iv)
//断言处理
stu_iv, err := iv.(Student)
if err {
fmt.Printf("stu_iv的类型%T\n", stu_iv)
}
}
func main() {
var stu Student = Student{
Name: "张三",
Age: 18,
Score: 80,
}
Test(stu)
}
//运行结果为:
类型是 main.Student
类别是 struct
c的类型是reflect.Kind
v的类型是reflect.Value
iv的类型main.Student
stu_iv的类型main.Student
2、ValueOf
获取变量值
reflect.valueof(x).Float()
reflect.valueof(x).Int()
reflect.valueof(x).String()
reflect.Valueof(x).Bool()
示例:类型转换。
package main
import (
"fmt"
"reflect"
)
func Test(i interface{}) {
v := reflect.ValueOf(i)
fmt.Printf("v的类型是%T\n", v)
//转换成指定类型
t := v.Int()
fmt.Printf("t的类型是%T\n", t)
}
func main() {
//类型不同的话会报错
var num int = 100
Test(num)
}
//运行结果为:
v的类型是reflect.Value
t的类型是int64
3、Value.Set
设置变量值
reflect.Value.SetFloat(),设置浮点数
reflect.value.SetInt(),设置整数
reflect.Value.SetString(),设置字符串
示例:
package main
import (
"fmt"
"reflect"
)
func Test(i interface{}) {
v := reflect.ValueOf(i)
//更新值需要value的地址,否则会保存,Elem()表示指针*
v.Elem().SetInt(100)
result := v.Elem().Int()
fmt.Printf("result类型为 %T, 值为 %d\n", result, result)
}
func main() {
var num int = 10
Test(&num)
}
//运行结果为:
result类型为 int64, 值为 100
4、结构体反射
示例:反射出结构体的属性和方法数量。
package main
import (
"fmt"
"reflect"
)
type Student struct {
Name string
Age int
Score float32
}
//结构方法
func (s Student) Run() {
fmt.Println("正在跑步。。。")
}
func (s Student) Sleep() {
fmt.Println("正在睡觉。。。")
}
//使用反射查看结构体字段数量和方法数量
func Test(i interface{}) {
v := reflect.ValueOf(i)
//类别判断
if v.Kind() != reflect.Struct {
fmt.Println("不是结构体")
return
}
//获取结构体中字段数量
stu_num := v.NumField()
fmt.Println("字段数量:", stu_num)
//获取结构体中方法数量
stu_meth := v.NumMethod()
fmt.Println("方法数量:", stu_meth)
}
func main() {
var stu Student = Student{
Name: "zhangsan",
Age: 18,
Score: 80,
}
Test(stu)
}
//运行结果为:
字段数量: 3
方法数量: 2
5、函数反射
Go 中函数是可以赋值给变量的。
示例:既然函数可以像普通的类型变量一样,那么在反射机制中就和不同的变量是一样的,在反射中函数和方法的类型(Type)都是reflect.Func,如果要调用函数,通过 Value 的Call() 方法。
package main
import (
"fmt"
"reflect"
)
func hello() {
fmt.Println("hello world")
}
func main() {
//反射使用函数
v := reflect.ValueOf(hello)
//类型判断是否属于reflect.func类型
if v.Kind() == reflect.Func {
fmt.Println("函数")
}
//反射调用函数
v.Call(nil) //Call中需要传入的是切片
}
//运行结果为:
函数
hello world
示例:
package main
import (
"fmt"
"reflect"
"strconv"
)
//反射调用传参和返回值函数
func Test(i int) string {
return strconv.Itoa(i)
}
func main() {
v := reflect.ValueOf(Test)
//定义参数切片
params := make([]reflect.Value, 1)
//切片元素赋值
params[0] = reflect.ValueOf(20)
//反射调函数
result := v.Call(params)
fmt.Printf("result的类型是 %T\n", result)
//[]reflect.Value切片转换string
s := result[0].Interface().(string)
fmt.Printf("s的类型是 %T ,值为 %s\n", s, s)
}
//运行结果为:
result的类型是 []reflect.Value
s的类型是 string ,值为 20
示例:获取结构体属性。
package main
import (
"fmt"
"reflect"
)
type Student struct {
Name string
Age int
Score float32
}
//反射获取结构体中定义的属性值和类型
func TestPrint(i interface{}) {
v := reflect.ValueOf(i)
//遍历结构体中所有属性
for i := 0; i < v.NumField(); i++ {
//输出属性值和类型
fmt.Printf("索引:%d,值:%v,类型:%v\n", i, v.Field(i), v.Field(i).Kind())
}
}
func main() {
//实例化
var stu Student = Student{
Name: "zhangsan",
Age: 20,
Score: 80,
}
TestPrint(stu)
}
//运行结果为:
索引:0,值:zhangsan,类型:string
索引:1,值:20,类型:int
索引:2,值:80,类型:float32
示例:更改结构体属性值。
package main
import (
"fmt"
"reflect"
)
type Student struct {
Name string
Age int
Score float32
}
//反射获取结构体中定义的属性值和类型
func TestPrint(i interface{}) {
v := reflect.ValueOf(i)
//遍历结构体中所有属性
for i := 0; i < v.NumField(); i++ {
//输出属性值和类型
fmt.Printf("索引:%d,值:%v,类型:%v\n", i, v.Field(i), v.Field(i).Kind())
}
}
//更新值
func SetVaule(i interface{}, name string) {
v := reflect.ValueOf(i)
vk := v.Kind()
//如果不是指针,而指针指向的类型是结构体,则报异常退出
if vk != reflect.Ptr && v.Elem().Kind() == reflect.Struct {
fmt.Println("type err")
return
}
//修改值
v.Elem().Field(0).SetString(name)
//遍历结构体中所有属性
for i := 0; i < v.Elem().NumField(); i++ {
//输出属性值和类型
fmt.Printf("索引:%d,值:%v,类型:%v\n", i, v.Elem().Field(i), v.Elem().Field(i).Kind())
}
}
func main() {
//实例化
var stu Student = Student{
Name: "zhangsan",
Age: 20,
Score: 80,
}
SetVaule(&stu, "lisi")
}
//运行结果为:
索引:0,值:lisi,类型:string
索引:1,值:20,类型:int
索引:2,值:80,类型:float32
示例:TAG原信息处理。
package main
import (
"encoding/json"
"fmt"
"reflect"
)
type Student struct {
Name string `json:"stu_name"`
Age int
Score float32
}
//反射获取结构体中定义的属性值和类型
func TestPrint(i interface{}) {
v := reflect.ValueOf(i)
//遍历结构体中所有属性
for i := 0; i < v.NumField(); i++ {
//输出属性值和类型
fmt.Printf("索引:%d,值:%v,类型:%v\n", i, v.Field(i), v.Field(i).Kind())
}
}
func main() {
//实例化
var stu Student = Student{
Name: "zhangsan",
Age: 20,
Score: 80,
}
TestPrint(stu)
fmt.Println("原信息---------------------")
result, _ := json.Marshal(stu)
fmt.Println("json原信息:", string(result))
fmt.Println("反射获取原信息--------------")
//反射获取类型属性
st := reflect.TypeOf(stu)
s := st.Field(0)
fmt.Printf("Name原信息名称:%s\n", s.Tag.Get("json"))
}
//运行结果为:
索引:0,值:zhangsan,类型:string
索引:1,值:20,类型:int
索引:2,值:80,类型:float32
原信息---------------------
json原信息: {"stu_name":"zhangsan","Age":20,"Score":80}
反射获取原信息--------------
Name原信息名称:stu_name
6、方法反射
反射中方法调用。函数和方法可以说其实本质上形同的,只不过方法与一个”对象“进行了”绑定“,方法是”对象“的一种行为,这种行为是对于这个”对象“的一系列操作,例如修改”对象“的某个属性。
示例:
package main
import (
"fmt"
"reflect"
"strconv"
)
//反射方法
type Student struct {
Name string
Age int
}
//方法
func (s *Student) SetName(name string) {
s.Name = name
}
func (s *Student) SetAge(age int) {
s.Age = age
}
func (s *Student) String() string {
return fmt.Sprintf("%p", s) + ",Name:" + s.Name + ",Age:" + strconv.Itoa(s.Age)
}
func main() {
//实例化
stu := &Student{"zhangsan", 19}
//反射获取值:指针方式
stuv := reflect.ValueOf(&stu).Elem()
fmt.Println("修改前:", stuv.MethodByName("String").Call(nil)[0])
//修改值
params := make([]reflect.Value, 1)
params[0] = reflect.ValueOf("lisi")
stuv.MethodByName("SetName").Call(params)
params[0] = reflect.ValueOf(20)
stuv.MethodByName("SetAge").Call(params)
fmt.Println("修改后:", stuv.MethodByName("String").Call(nil)[0])
}
//运行结果为:
修改前: 0xc000004078,Name:zhangsan,Age:19
修改后: 0xc000004078,Name:lisi,Age:20
示例:
package main
//方式二:反射使用method
import (
"fmt"
"reflect"
"strconv"
)
//反射方法
type Student struct {
Name string
Age int
}
//方法
func (s *Student) SetName(name string) {
s.Name = name
}
func (s *Student) SetAge(age int) {
s.Age = age
}
func (s *Student) String() string {
return fmt.Sprintf("%p", s) + ",Name:" + s.Name + "Age:" + strconv.Itoa(s.Age)
}
func main() {
//实例化
stu := &Student{"zhangsan", 19}
//反射获取值:指针方式
stuv := reflect.ValueOf(&stu).Elem()
//索引调用方法
fmt.Println("修改前:", stuv.Method(2).Call(nil)[0])
params := make([]reflect.Value, 1)
params[0] = reflect.ValueOf(20)
stuv.Method(0).Call(params)
params[0] = reflect.ValueOf("lisi")
stuv.Method(1).Call(params)
fmt.Println("修改后:", stuv.Method(2).Call(nil)[0])
}
//运行结果为:
修改前: 0xc000004078,Name:zhangsanAge:19
修改后: 0xc000004078,Name:lisiAge:20