本文主要是介绍反射,对大家解决编程问题具有一定的参考价值,需要的程序猿们随着小编来一起学习吧!
反射
package main
import (
"fmt"
"reflect"
)
type Student struct {
name string
}
type MyInt int
func main() {
/*Type的主要方法
Kind()返回一个常量,表示具体类型的底层类型
Elem()返回指针、数组、切片、字典、通道等类型
*/
/*Value的主要方法
Kind()返回一个常量,表示具体类型的底层类型
Type()返回具体类型所对应的reflect.Type(静态类型)
*/
/*
在go语言中,静态类型就是变量声明时赋予的类型,也就是在发射中reflect.Type对应的值,而Kind()对应的是基础类型。
Kind()大概会返回切片、字典、指针、结构体、接口、字符串、数组、函数、整形或其他基础类型。如下面代码中Kind()
返回结构体:fmt.Println(p.Kind()),而Type()返回静态类型名Student:fmt.Println(p.Type()).MyInt是静态类型
而int是它的基础类型。
Type()返回的是静态类型,而Kind()返回的是基础类型
*/
var a int = 9
v := reflect.ValueOf(a) //返回Value类型对象,值为9
t := reflect.TypeOf(a) //返回Type类型对象,值为int
fmt.Println(v, t, v.Type(), v.Kind(), t.Kind()) //Kind()返回底层基础类型
/*
9 int int int int
*/
var mi MyInt = 99
mv := reflect.ValueOf(mi) //返回Value类型对象,值为99
mt := reflect.TypeOf(mi) // 返回Type类型对象,值为MyInt
fmt.Println(mv, mt, mv.Type(), mv.Kind(), mt.Kind()) //Kind()返回底层基础类型
/*
99 main.MyInt main.MyInt int int
*/
var b [5]int = [5]int{5, 6, 7, 8}
fmt.Println(reflect.TypeOf(b), reflect.TypeOf(b).Kind(), reflect.TypeOf(b).Elem())
/*
[5]int array int
*/
var Pupil Student
p := reflect.ValueOf(Pupil) //使用ValueOf获得结构体的Value对象
fmt.Println(p.Type(), p.Kind())
/*
main.Student struct
*/
}
反射的应用
/*反射的应用
(1)通过反射可以修改对象
通过反射可以修改对象,但对象必须是可寻址的(addressable)。简单说,如果想通过反射修改对象,就需要把想修改
对象的指针传递过来。如果对象不能被寻址,那就是不可写的。可写性是反射类型变量的一个属性,但不是所有反射类型
变量都拥有这个属性,所以通过反射修改原对象,需要判断其可写性,也就是可寻址。
实际上要修改的是指针指向的数据,需要调用Value类型的Elem()方法。Elem()方法能够对指针进行间接引用,将结果
存储到reflect.Value类型对象中。
v := reflect.Value.Elem() //表示获取原始值对应的反射对象
通过CanSet()方法来判断原始反射对象v reflect.Value是否可写,CanAddr()方法判断它是否可被取地址。这里的v是通过
Elem()得到的。CanSet()和CanAddr()这两个方法的签名如下:
func (v Value)CanAddr()
func (v Value)CanSet() bool
*/
var a int = 9
v := reflect.ValueOf(a) // 返回Value类型对象,值为9
t := reflect.TypeOf(a) // 返回Type类型对象,值为int
fmt.Println(v.Type(), t.Kind(), reflect.ValueOf(&a).Elem())
/*
int int 9
*/
fmt.Println(reflect.ValueOf(a).CanSet(), reflect.ValueOf(a).CanAddr())
/*
false false
*/
fmt.Println(reflect.ValueOf(&a).CanSet(), reflect.ValueOf(&a).CanAddr())
/*
false false
*/
pa := reflect.ValueOf(&a).Elem() // reflect.Value.Elem()表示获取原始值对应的反射对象
fmt.Println(pa.CanSet(), pa.CanAddr())
/*
true true
*/
pa.SetInt(100)
fmt.Println(pa)
/*
100
*/
var Pupil Student = Student{"jim", 8}
Pupilv := reflect.ValueOf(Pupil) //使用ValueOf()获取结构体的Value对象
fmt.Println(Pupilv.Type(), Pupilv.Kind())
/*
main.Student struct
*/
p := reflect.ValueOf(&Pupil).Elem() //获取原始值对应的反射对象
fmt.Println(p.CanSet(), p.CanAddr())
/*
true true
*/
// p.Field(0).SetString("Mike") 报错,原因是字段需要大写(需可以被外部包调用)
p.Field(1).SetInt(10)
fmt.Println(p)
/*
{jim 10}
*/
/*
要通过反射的方式来修改对象,重点是通过方法Elem()获取原始值对应的反射对象。虽然反射可以越过go语言的导出规则
的限制读取结构中未导出的成员,但不能修改他们。因为一个结构体中只有被导出(可被外部包调用)的字段才是可写的。
reflect.ValueOf(&a)得到的是原始变量a的指针地址,这个指针地址再通过Elem()方法得到反射对象。
*/
/*
结构体中有tag标签,通风反射可获取结构体成员变量的tag信息。
*/
var s Student = Student{"joke", 18}
setStudent := reflect.ValueOf(&s).Elem()
sSAge, _ := setStudent.Type().FieldByName("Age")
fmt.Println(sSAge.Tag.Get("json"))
/*
years
*/
/*
(2)通过反射可以创建基础类型和用户自定义类型变量
除了可以通过反射创建基础类型和用户自定义类型,还可以使用反射来创建切片,字典,通道,甚至包括函数类型。常见函数有:
reflect.Makeslice(),reflect.Makemap()和reflect.Makechan()
*/
/*
想创建变量,需要先确定类型。下面的代码中根据reflect.Type(t)得到t的静态类型,接着使用reflect.New(vartype)生成了新变量。
新变量通过方法Elem()获取的反射对象来设置变量值。最后使用Elem().interface()来反引用reflect的指针,得到新变量的值。
*/
/*
t := 9
// 反射创建int变量
varType := reflect.TypeOf(t)
v1 := reflect.New(varType)
v1.Elem().SetInt(1)
varNew := v1.Elem().Interface()
fmt.Printf("指针:%d, 值:%d\n", v1, varNew)
// 反射创建map slice
newSlice := make([]int, 5)
newmap := make(map[string]int)
sliceType := reflect.TypeOf(newSlice)
mapType := reflect.TypeOf(newmap)
// 创建新值
ReflectSlice := reflect.MakeSlice(sliceType, 5, 5)
Reflectmap := reflect.MakeMap(mapType)
// 使用新创建的变量
V := 99
SliceV := reflect.ValueOf(V)
ReflectSlice = reflect.Append(ReflectSlice, SliceV)
intSlice := ReflectSlice.Interface().([]int)
fmt.Println("Slice:", intSlice)
Key := "Rose"
Value := 999
MapKey := reflect.ValueOf(Key)
MapValue := reflect.ValueOf(Value)
Reflectmap.SetMapIndex(MapKey, MapValue)
mapStringInt := Reflectmap.Interface().(map[string]int)
fmt.Println("Map:", mapStringInt)
/*
指针:824633811096, 值:1
Slice: [0 0 0 0 0 99]
Map: map[Rose:999]
*/
}
通过反射机制,能对一个结构体类型的大致结构如方法、字段的情况有较为全面的了解
package main
import (
"fmt"
"reflect"
)
type ss struct {
int
string
bool
float64
}
func (s ss) Method(i int) string { return "结构体方法1" }
func (s *ss) Method2(i int) string { return "结构体方法2" }
var (
structValue = ss{ // 结构体
20,
"结构体",
false,
64.0,
}
)
func main() {
// 通过反射机制,能对一个结构体类型的大致结构如方法、字段的情况有较为全面的了解
fmt.Println("-------------引用------------")
v := reflect.ValueOf(&structValue)
fmt.Println(v.String()) // 反射值的字符串形式
fmt.Println(v.Type()) //反射值的类型
fmt.Println(v.Kind()) //反射值的类别
fmt.Println(v.CanAddr()) //是否可以获取地址
fmt.Println(v.CanSet()) //是否可以修改
if v.CanAddr() {
fmt.Println(v.Addr()) //获取地址
fmt.Println(v.UnsafeAddr()) // 获取自由地址
}
// 获取方法数量
fmt.Println("可用方法数量:", v.NumMethod())
if v.NumMethod() > 0 {
i := 0
for ; i < v.NumMethod()-1; i++ {
fmt.Println(v.Method(i).String())
}
fmt.Println(v.Method(i).String())
// 通过名称获取方法
fmt.Println(v.MethodByName("Method1").String())
fmt.Println(v.MethodByName("Method2").String())
}
fmt.Println("------------------值变量------------------")
v = reflect.ValueOf(structValue)
fmt.Println(v.String()) //反射值的字符串形式
fmt.Println(v.Type()) // 反射值的类型
fmt.Println(v.Kind()) // 反射值的类别
fmt.Println(v.CanAddr()) //是否可以获取地址
fmt.Println(v.CanSet()) //是否可以修改
if v.CanAddr() {
fmt.Println(v.Addr()) //获取地址
fmt.Println(v.UnsafeAddr()) //获取自由地址
}
// 获取方法数量
fmt.Println(v.NumMethod())
fmt.Println(reflect.ValueOf(&structValue).NumMethod())
if v.NumMethod() > 0 {
i := 0
for ; i < v.NumMethod()-1; i++ {
fmt.Println(v.Method(i).String())
}
fmt.Println(v.Method(i).String())
// 通过名称获取方法
fmt.Println(v.MethodByName("Method1").String())
fmt.Println(v.MethodByName("Method2").String())
}
switch v.Kind() {
// 结构体
case reflect.Struct:
fmt.Println("-----------结构体----------")
// 获取字段个数
fmt.Println(v.NumField())
if v.NumField() > 0 {
var i int
//遍历结构体字段
for i = 0; i < v.NumField()-1; i++ {
field := v.Field(i) //获取结构体字段
fmt.Println(field.Type(), field.String())
}
field := v.Field(i) // 获取结构体字段
fmt.Println(field.Type(), field.String())
// 通过名称查找字段
if v := v.FieldByName("ptr"); v.IsValid() {
fmt.Println(v.Type().Name())
}
// 通过函数查找字段
v := v.FieldByNameFunc(func(s string) bool { return len(s) > 3 })
if v.IsValid() {
fmt.Println(v.Type().Name())
}
}
}
}
/*
-------------引用------------
<*main.ss Value>
*main.ss
ptr
false
false
可用方法数量: 2
<func(int) string Value>
<func(int) string Value>
<invalid Value>
<func(int) string Value>
------------------值变量------------------
<main.ss Value>
main.ss
struct
false
false
1
2
<func(int) string Value>
<invalid Value>
<invalid Value>
-----------结构体----------
4
int <int Value>
string 结构体
bool <bool Value>
float64 <float64 Value>
*/
这篇关于反射的文章就介绍到这儿,希望我们推荐的文章对大家有所帮助,也希望大家多多支持为之网!