Go語言基礎(chǔ)反射示例詳解

概述
在程序運(yùn)行期對程序動態(tài)的進(jìn)行訪問和修改
reflect godoc: https://golang.org/pkg/reflect/
reflect包有兩個數(shù)據(jù)類型:
Type:數(shù)據(jù)類型 【reflect.TypeOf():是獲取Type的方法】
Value:值的類型【reflect.ValueOf():是獲取Value的方法】
語法
一、基本操作
獲取變量類型
func TypeOf(i interface{}) Type //Type是interface{}的別名
例子
reflect.TypeOf(10) //int
reflect.TypeOf(struct{ age int }{10}) //struct { age int }
獲取變量的種類
reflect.TypeOf(struct{ age int }{10}).Kind() //reflect.Struct
reflect.ValueOf("hello word").Kind() //reflect.String
獲取變量值
func ValueOf(i interface{}) Value //value是struct {}別名
例子
reflect.ValueOf("hello word") //hello word
reflect.ValueOf(struct{ age int }{10}) //{10}
二、修改目標(biāo)對象
修改普通類型
str := "hello word"
reflect.ValueOf(&str).Elem().SetString("張三")
修改結(jié)構(gòu)體
//第一步:ValueOf():傳入一個變量的地址,返回是變量的地址 Elem():返回的是變量的原始值
elem:=reflect.ValueOf(&變量名).Elem()
//第二步 FieldByName():傳入結(jié)構(gòu)體字段名稱 SetString():傳入你要修改的變量值
elem.FieldByName("Name").SetString("李四")
//定義一個User結(jié)構(gòu)體
type User struct {
Name string
Age int
}
user := User{Name: "張三", Age: 10}
//Elem() 獲取user原始的值
elem := reflect.ValueOf(&user).Elem()
//FieldByName() 通過Name返回具有給定名稱的結(jié)構(gòu)字段 通過SetString 修改原始的值
elem.FieldByName("Name").SetString("李四")
elem.FieldByName("Age").SetInt(18)
三、動態(tài)調(diào)用方法
無參方法
//MethodByName():傳方法名,方法名必須大小 Call():方法的形參
reflect.ValueOf(變量名).MethodByName(方法名).Call([]reflect.Value{})
reflect.ValueOf(變量名).MethodByName(方法名).Call(make([]reflect.Value, 0))
type User struct {
Name string `json:"name" name:"張三"`
Age int
}
func (_ User) Say() {
fmt.Println("user 說話")
}
user := User{Name: "張三", Age: 10}
reflect.ValueOf(&user).MethodByName("Say").Call([]reflect.Value{})
reflect.ValueOf(user).MethodByName("Say").Call(make([]reflect.Value, 0))
有參方法
reflect.ValueOf(變量名).MethodByName(方法名).Call([]reflect.Value{reflect.ValueOf("該說話了"), reflect.ValueOf(1)})
type User struct {
Name string `json:"name" name:"張三"`
Age int
}
func (_ User) Say() {
fmt.Println("user 說話")
}
user := User{Name: "張三", Age: 10}
reflect.ValueOf(user).MethodByName("SayContent").Call([]reflect.Value{reflect.ValueOf("該說話了"), reflect.ValueOf(1)})
總結(jié)
反射調(diào)用struct的方法必須是公有的
反射調(diào)用無參方法時(shí)必修傳 nil 或者 []reflect.Value{}
示例
package main
import (
"fmt"
"reflect"
)
func main() {
//1. 獲取變量類型
fmt.Println("獲取變量類型")
fmt.Println(reflect.TypeOf(10)) //int
fmt.Println(reflect.TypeOf(10.0)) //float64
fmt.Println(reflect.TypeOf(struct{ age int }{10})) //struct { age int }
fmt.Println(reflect.TypeOf(map[string]string{"a": "a"})) //map[string]string
fmt.Println("")
//2. 獲取變量值
fmt.Println("獲取變量值")
fmt.Println(reflect.ValueOf("hello word")) //hello word
fmt.Println(reflect.ValueOf(struct{ age int }{10})) //{10}
fmt.Println(reflect.TypeOf(struct{ age int }{10}).Kind()) //struct
//類型判斷
if t := reflect.TypeOf(struct{ age int }{10}).Kind(); t == reflect.Struct {
fmt.Println("是結(jié)構(gòu)體")
} else {
fmt.Println("不是結(jié)構(gòu)體")
}
//修改目標(biāo)對象
str := "hello word"
//普通變量修改
reflect.ValueOf(&str).Elem().SetString("張三")
fmt.Println(str)
//結(jié)構(gòu)體變量修改
user := User{Name: "張三", Age: 10}
//Elem() 獲取user原始的值
elem := reflect.ValueOf(&user).Elem()
//FieldByName() 通過Name返回具有給定名稱的結(jié)構(gòu)字段 通過SetString 修改原始的值
elem.FieldByName("Name").SetString("李四")
elem.FieldByName("Age").SetInt(18)
fmt.Println(user)
//獲取結(jié)構(gòu)體的標(biāo)簽的值
fmt.Println(reflect.TypeOf(&user).Elem().Field(0).Tag.Get("name"))
//調(diào)用無參方法
reflect.ValueOf(&user).MethodByName("Say").Call([]reflect.Value{})
reflect.ValueOf(user).MethodByName("Say").Call(make([]reflect.Value, 0))
//調(diào)用有參方法
reflect.ValueOf(user).MethodByName("SayContent").Call([]reflect.Value{reflect.ValueOf("該說話了"), reflect.ValueOf(1)})
//調(diào)用本地的方法
reflect.ValueOf(Hello).Call([]reflect.Value{})
reflect.ValueOf(Hello).Call(nil)
fmt.Printf("%#v\n", reflect.TypeOf(user).Field(0))
}
func Hello() {
fmt.Println("hello")
}
type Person struct {
Name string
}
type User struct {
Person // //反射會將匿名字段作為一個獨(dú)立字段來處理
Name string `json:"name" name:"張三"`
Age int
}
func (_ User) Say() {
fmt.Println("user 說話")
}
func (_ User) SayContent(content string, a int) {
fmt.Println("user", content, a)
}
以上就是Go語言基礎(chǔ)反射示例詳解的詳細(xì)內(nèi)容,更多關(guān)于Go語言反射的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Go Gin框架中的binding驗(yàn)證器使用小結(jié)
Gin框架中的binding驗(yàn)證器為我們提供了簡便的數(shù)據(jù)綁定和驗(yàn)證功能,通過合理使用binding和validate標(biāo)簽,我們可以確保API接口的數(shù)據(jù)合法性和完整性,這篇文章主要介紹了Go Gin框架中的binding驗(yàn)證器使用指南,需要的朋友可以參考下2024-07-07
Go語言調(diào)用ffmpeg-api實(shí)現(xiàn)音頻重采樣
最近對golang處理音視頻很感興趣,對golang音視頻常用庫goav進(jìn)行了一番研究。自己寫了一個wav轉(zhuǎn)采樣率的功能。給大家分享一下,中間遇到了不少坑,解決的過程中還是蠻有意思的,希望大家能喜歡2022-12-12
golang的httpserver優(yōu)雅重啟方法詳解
這篇文章主要給大家介紹了關(guān)于golang的httpserver優(yōu)雅重啟的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。2018-03-03
Golang操作DuckDB實(shí)戰(zhàn)案例分享
DuckDB是一個嵌入式SQL數(shù)據(jù)庫引擎,它與眾所周知的SQLite非常相似,但它是為olap風(fēng)格的工作負(fù)載設(shè)計(jì)的,DuckDB支持各種數(shù)據(jù)類型和SQL特性,憑借其在以內(nèi)存為中心的環(huán)境中處理高速分析的能力,它迅速受到數(shù)據(jù)科學(xué)家和分析師的歡迎,在這篇博文中,我們將探索在Go中使用DuckDB2025-01-01
Golang?int函數(shù)使用實(shí)例全面教程
這篇文章主要為大家介紹了Golang?int函數(shù)使用實(shí)例全面教程詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-10-10

