Golang 反射reflection


反射reflection

  • 反射可大大提高程序的靈活性,使得interface{}有更大的發揮余地
  • 反射使用TypeOf和ValueOf函數從接口中獲取目標對象信息
  • 反射會將匿名字段作為獨立字段(匿名字段本質)
  • 想要利用反射修改對象狀態,前提是interface.data是settable,即pointer-interface
  • 通過反射可以“動態”調用方法

對某一個struct進行反射的基本操作

package main

import (
	"fmt"
	"reflect"
)

type User struct {
	Id   int
	Name string
	Age  int
}

func (u User) Hello() {
	fmt.Println("Hello world!")
}
func Info(o interface{}) {
	t := reflect.TypeOf(o)         //反射使用 TypeOf 和 ValueOf 函數從接口中獲取目標對象信息
	fmt.Println("Type:", t.Name()) //調用t.Name方法來獲取這個類型的名稱

	v := reflect.ValueOf(o) //打印出所包含的字段
	fmt.Println("Fields:")
	for i := 0; i < t.NumField(); i++ { //通過索引來取得它的所有字段,這里通過t.NumField來獲取它多擁有的字段數量,同時來決定循環的次數
		f := t.Field(i)               //通過這個i作為它的索引,從0開始來取得它的字段
		val := v.Field(i).Interface() //通過interface方法來取出這個字段所對應的值
		fmt.Printf("%6s:%v =%v\n", f.Name, f.Type, val)
	}
	for i := 0; i < t.NumMethod(); i++ { //這里同樣通過t.NumMethod來獲取它擁有的方法的數量,來決定循環的次數
		m := t.Method(i)
		fmt.Printf("%6s:%v\n", m.Name, m.Type)

	}
}
func main() {
	u := User{1, "Jack", 23}
	Info(u)
}

判斷傳入的類型是否是我們想要的類型

package main

import (
	"fmt"
	"reflect"
)

type User struct {
	Id   int
	Name string
	Age  int
}

func (u User) Hello() {
	fmt.Println("Hello world!")
}
func Info(o interface{}) {
	t := reflect.TypeOf(o)                  //反射使用 TypeOf 和 ValueOf 函數從接口中獲取目標對象信息
	fmt.Println("Type:", t.Name())          //調用t.Name方法來獲取這個類型的名稱
	if k := t.Kind(); k != reflect.Struct { //通過kind方法判斷傳入的類型是否是我們需要反射的類型
		fmt.Println("xx")
		return
	}
	v := reflect.ValueOf(o) //打印出所包含的字段
	fmt.Println("Fields:")
	for i := 0; i < t.NumField(); i++ { //通過索引來取得它的所有字段,這里通過t.NumField來獲取它多擁有的字段數量,同時來決定循環的次數
		f := t.Field(i)               //通過這個i作為它的索引,從0開始來取得它的字段
		val := v.Field(i).Interface() //通過interface方法來取出這個字段所對應的值
		fmt.Printf("%6s:%v =%v\n", f.Name, f.Type, val)
	}
	for i := 0; i < t.NumMethod(); i++ { //這里同樣通過t.NumMethod來獲取它擁有的方法的數量,來決定循環的次數
		m := t.Method(i)
		fmt.Printf("%6s:%v\n", m.Name, m.Type)

	}
}
func main() {
	u := User{1, "Jack", 23}
	Info(u)
}

反射 匿名或嵌入字段

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, "Jack", 12}, Title: "123"}
	t := reflect.TypeOf(m)
	fmt.Printf("%#v\n", t.Field(0))                   //#號會將reflect的struct的詳情頁打印出來,可以看出來這是一個匿名字段
	fmt.Printf("%#v \n", t.FieldByIndex([]int{0, 0})) //此時 我們就可以將User當中的ID取出來,這里面需要傳進方法中的是一個int類型的slice,User相對於manager索引是0,id相對於User索引也是0
	fmt.Printf("%v \n", t.FieldByIndex([]int{0, 1}))
	v := reflect.ValueOf(m)
	fmt.Printf("%#v\n", v.Field(0))
}

通過反射修改struct中的內容

package main

import (
	"fmt"
	"reflect"
)

func main() {
	x := 123
	v := reflect.ValueOf(&x)   
	//傳遞指針才能修改
	v.Elem().SetInt(999)
	fmt.Println(x)
}
PS G:\mygo\src\mytest> go run .\temp10.go
999
package main

import (
	"fmt"
	"reflect"
)

type User struct {
	Id   int
	Name string
	Age  int
}

func main() {
	u := User{1, "Tom", 12}
	Set(&u)
	fmt.Println(u)

}

func Set(o interface{}) {
	v := reflect.ValueOf(o)
	if v.Kind() == reflect.Ptr && !v.Elem().CanSet() {
		fmt.Println("xxx")
		return
	} else {
		v = v.Elem()
	}
	f := v.FieldByName("Name")
	if !f.IsValid() {
		fmt.Println("xiugaishibai")
	}
	if f.Kind() == reflect.String {
		f.SetString("jACK")
	}

}

通過發射進行方法的調用 動態調用方法

package main

import (
	"fmt"
	"reflect"
)

type User struct {
	Id   int
	Name string
	Age  int
}

func (u User) Hello(name string) {
	fmt.Println("Hello", name, "My name is", u.Name)
}

func main() {
	u := User{1, "OK", 12}
	v := reflect.ValueOf(u)
	mv := v.MethodByName("Hello")
	args := []reflect.Value{reflect.ValueOf("JOE")}
	mv.Call(args)
}


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM