反射的引入
【1】反射可以做什么?
1) 反射可以在運行時動態(tài)獲取變量的各種信息,比如變量的類型,類別等信息
2) 如果是結(jié)構(gòu)體變量,還可以獲取到結(jié)構(gòu)體本身的信息(包括結(jié)構(gòu)體的字段、方法)
3) 通過反射,可以修改變量的值,可以調(diào)用關(guān)聯(lián)的方法。
4) 使用反射,需要import ("reflect")
【2】反射相關(guān)的函數(shù)
1) reflect.TypeOf(變量名),獲取變量的類型,.Type類型
2) reflect.ValueOf(變量名),獲取變量的值,返回reflect.Value類型(reflect.Value是一個結(jié)構(gòu)體類型),通過reflect.Value,可以獲取到關(guān)于該變量的很多信息。
對基本數(shù)據(jù)類型反射
【1】反射相關(guān)的函數(shù)
1) reflect.TypeOf(變量名),獲取變量的類型,返回reflect.Type類型
2) reflect.ValueOf(變量名),獲取變量的值,返回reflect.Value類型(reflect.Value是一個結(jié)構(gòu)體類型),通過reflect.Value,可以獲取到關(guān)于該變量的很多信息。
【2】代碼:
package mAIn
import (
"fmt"
"reflect"
)
//利用一個函數(shù),函數(shù)的參數(shù)定義為空接口:
//空接口沒有任何方法,所以可以理解為所有類型都實現(xiàn)了空接口,也可以理解為我們可以把任何一個變量賦給空接口。
func testReflect(i interface{}) {
//1.調(diào)用TypeOf函數(shù),返回reflect.Type類型數(shù)據(jù):
reType := reflect.TypeOf(i)
fmt.Println("reType:",reType)
fmt.Printf("reType的具體類型是:%T n",reType)
//2.調(diào)用ValueOf函數(shù),返回reflect.Value類型數(shù)據(jù):
reValue := reflect.ValueOf(i)
fmt.Println("reValue",reValue)
fmt.Printf("reValue的具體類型是:%T n",reValue)
// num1 := 100
//如果真想獲取reValue的數(shù)值,要調(diào)用Int()方法:返回v持有的有符號整數(shù)
num2 := 80 + reValue.Int()
fmt.Println(num2)
//reValue轉(zhuǎn)成空接口:
i2 := reValue.Interface()
n := i2.(int)
n2 := n + 30
fmt.Println(n2)
}
func main() {
var num int = 100;
testReflect(num)
}
對結(jié)構(gòu)體類型反射
【1】反射相關(guān)的函數(shù)
1) reflect.TypeOf(變量名),獲取變量的類型,返回reflect.Type類型
2) reflect.ValueOf(變量名),獲取變量的值,返回reflect.Value類型(reflect.Value是一個結(jié)構(gòu)體類型),通過reflect.Value,可以獲取到關(guān)于該變量的很多信息。
【2】代碼:
package main
import (
"fmt"
"reflect"
)
//利用一個函數(shù),函數(shù)的參數(shù)定義為空接口:
//空接口沒有任何方法,所以可以理解為所有類型都實現(xiàn)了空接口,也可以理解為我們可以把任何一個變量賦給空接口。
func testReflect(i interface{}) {
//1.調(diào)用TypeOf函數(shù),返回reflect.Type類型數(shù)據(jù):
reType := reflect.TypeOf(i)
fmt.Println("reType:",reType)
fmt.Printf("reType的具體類型是:%T n",reType)
//2.調(diào)用ValueOf函數(shù),返回reflect.Value類型數(shù)據(jù):
reValue := reflect.ValueOf(i)
fmt.Println("reValue",reValue)
fmt.Printf("reValue的具體類型是:%T n",reValue)
// num1 := 100
//如果真想獲取reValue的數(shù)值,要調(diào)用Int()方法:返回v持有的有符號整數(shù)
num2 := 80 + reValue.Int()
fmt.Println(num2)
//reValue轉(zhuǎn)成空接口:
i2 := reValue.Interface()
n := i2.(int)
n2 := n + 30
fmt.Println(n2)
}
func main() {
var num int = 100;
testReflect(num)
}
獲取變量的類別
【1】獲取變量的類別:兩種方式:
(1)reflect.Type.Kind()
(2)reflect.Value.Kind()
【2】Kind的值是常量值:
【3】代碼:
package main
import (
"fmt"
"reflect"
)
type Student struct {
Name string
Age int
}
func testReflect(i interface{}) {
reType := reflect.TypeOf(i)
reValue := reflect.ValueOf(i)
//獲取變量的類別:
//(1)reType.Kind()
k1 := reType.Kind()
fmt.Println(k1)
//(2)reValue.Kind()
k2 := reValue.Kind()
fmt.Println(k2)
//獲取變量的類型:
i2 := reValue.Interface()
n,flag := i2.(Student)
if flag == true {
fmt.Printf("結(jié)構(gòu)體的類型是:%T",n)
}
}
func main() {
stu := Student{
Name: "菜園子",
Age: 18,
}
testReflect(stu)
}
【4】Type和 Kind 的區(qū)別
Type是類型, Kind是類別,Type和Kind 可能是相同的,也可能是不同的.
比如:var num int = 10 num的Type是int , Kind也是int
比如:var stu Studentstu的 Type是 pkg1.Student , Kind是struct
通過反射修改變量
修改基本數(shù)據(jù)類型的值:
package main
import (
"fmt"
"reflect"
)
func testReflect(i interface{}) {
reValue := reflect.ValueOf(i)
//通過SetInt()來改變值:
reValue.Elem().SetInt(50)
}
func main() {
var num int = 100
//傳入指針地址
testReflect(&num)
fmt.Println(num)
}
通過反射操作結(jié)構(gòu)體的屬性和方法
【1】代碼:(熟知API)
package main
import (
"fmt"
"reflect"
)
type Student struct {
Name string
Age int
}
//給結(jié)構(gòu)體綁定方法:
func (s Student) Print() {
fmt.Println("調(diào)用了Print()方法")
fmt.Println("學(xué)生的名字是:",s.Name)
}
func (s Student) GetSum(n1,n2 int) int{
fmt.Println("調(diào)用了GetSum()方法")
return n1 + n2
}
func (s Student) Set(name string,age int) {
s.Name = name
s.Age = age
}
//定義函數(shù)操作結(jié)構(gòu)體進行反射操作:
func TestStudentStruct(a interface{}) {
//a轉(zhuǎn)成reflect.Value類型:
val := reflect.ValueOf(a)
fmt.Println(val)
//通過reflect.Value類型操作結(jié)構(gòu)體內(nèi)部的字段:
n1 := val.NumField()
fmt.Println(n1)
//遍歷-獲取具體的字段:
for i := 0; i < n1; i++ {
fmt.Printf("第%d個字段的值是:%v n",i+1,val.Field(i))
}
//通過reflect.Value類型操作結(jié)構(gòu)體內(nèi)部的方法:
n2 := val.NumMethod()
fmt.Println(n2)
//調(diào)用Print()方法:
//調(diào)用方法,方法的首字母必須大寫才能有對應(yīng)的反射的訪問權(quán)限
//方法的順序按照ASCII的順序排列的,a,b,c,,,,,,索引:0,1,2,,,
val.Method(1).Call(nil)
//調(diào)用GetSum方法:
//定義Value的切片:
var params []reflect.Value
params = Append(params,reflect.ValueOf(10))
params = append(params,reflect.ValueOf(20))
result := val.Method(0).Call(params)
fmt.Println("GetSum方法的返回值是:",result[0].Int())
}
func main() {
stu := Student{
Name: "菜園子",
Age: 19,
}
TestStudentStruct(stu)
}
通過反射修改變量
【1】代碼:
package main
import (
"fmt"
"reflect"
)
type Student struct{
Name string
Age int
}
func (s Student) Print() {
fmt.Println("調(diào)用了Print()方法")
fmt.Println("學(xué)生的姓名是:",s.Name)
}
func (s Student) GetSum(n1,n2 int) int {
fmt.Println("調(diào)用了GetSum()方法")
return n1 + n2
}
func (s Student) Set(name string,age int) {
s.Name = name
s.Age = age
}
func TestReflectStuct(a interface{}) {
//a轉(zhuǎn)成reflect.Value類型:
val := reflect.ValueOf(a)
fmt.Println(val)
n := val.Elem().NumField()
fmt.Println(n)
//修改字段的值:
val.Elem().Field(0).SetString("Idea")
}
func main() {
stu := Student {
Name: "菜園子",
Age: 19,
}
TestReflectStuct(&stu)
fmt.Println(stu)
}
參考源碼:golang-demo: golang學(xué)習(xí)
參考博客:cyz