<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
反射是程式在執行期間獲取變數的型別和值、或者執行變數的方法的能力。
Golang反射包中有兩對非常重要的函數和型別,兩個函數分別是:
reflect.TypeOf能獲取型別資訊reflect.Type;
reflect.ValueOf 能獲取資料的執行時表示reflect.Value;
Golang是一門靜態型別的語言,反射是建立在型別之上的。
通過reflect.TypeOf()函數可以獲得任意值的型別資訊。
諸如int32, slice, map以及通過type關鍵詞自定義的型別。
種類Kind可以理解為型別的具體分類。如int32、type MyInt32 int32是兩種不同型別,但都屬於int32這個種類。
使用 reflect.TypeOf()獲取變數型別以及種類。
func main() { type MyInt32 int32 a := MyInt32(1) b := int32(1) fmt.Printf("reflect.TypeOf(a):%v Kind:%vn", reflect.TypeOf(a), reflect.TypeOf(a).Kind()) fmt.Printf("reflect.TypeOf(b):%v Kind:%vn", reflect.TypeOf(b), reflect.TypeOf(b).Kind()) }
程式碼輸出如下,由此可以看出int32、type MyInt32 int32是兩種不同型別,但都屬於int32這個種類。
$ go run main.go
reflect.TypeOf(a):main.MyInt32 Kind:int32
reflect.TypeOf(b):int32 Kind:int32
種類定義點選檢視
// A Kind represents the specific kind of type that a Type represents. // The zero Kind is not a valid kind. type Kind uint const ( Invalid Kind = iota Bool Int Int8 Int16 Int32 Int64 Uint Uint8 Uint16 Uint32 Uint64 Uintptr Float32 Float64 Complex64 Complex128 Array Chan Func Interface Map Pointer Slice String Struct UnsafePointer )
// Elem returns a type's element type.
// It panics if the type's Kind is not Array, Chan, Map, Pointer, or Slice.
Elem() Type
部分情況我們需要獲取指標指向元素的型別、或者slice元素的型別,可以reflect.Elem()函數獲取。
func main() { type myStruct struct { } a := &myStruct{} typeA := reflect.TypeOf(a) fmt.Printf("TypeOf(a):%v Kind:%vn", typeA, typeA.Kind()) fmt.Printf("TypeOf(a).Elem():%v Elem().Kind:%vn", typeA.Elem(), typeA.Elem().Kind()) s := []int64{} typeS := reflect.TypeOf(s) fmt.Printf("TypeOf(s):%v Kind:%vn", typeS, typeS.Kind()) fmt.Printf("TypeOf(s).Elem():%v Elem().Kind:%vn", typeS.Elem(), typeS.Elem().Kind()) }
程式碼輸出如下,由此可以看出,通過reflect.Elem()函數可以獲取參照指向資料的型別。
$ go run main.go
TypeOf(a):*main.myStruct Kind:ptr
TypeOf(a).Elem():main.myStruct Elem().Kind:struct
TypeOf(s):[]int64 Kind:slice
TypeOf(s).Elem():int64 Elem().Kind:int64
通過NumField獲取成員數量,Field通過下標存取成員的型別資訊StructField,包括成員名稱、型別、Tag資訊等。
func main() { type secStruct struct { Cnt []int64 } type myStruct struct { Num int `json:"num_json" orm:"column:num_orm"` Desc string `json:"desc_json" orm:"column:desc_orm"` Child secStruct } s := myStruct{} typeS := reflect.TypeOf(s) // 成員數量 fmt.Printf("NumField:%v n", typeS.NumField()) // 每個成員的資訊 包括名稱、型別、Tag for i := 0; i < typeS.NumField(); i++ { // 通過下標存取成員 fmt.Printf("Field(%v):%+vn", i, typeS.Field(i)) } // 通過名稱存取成員 field, ok := typeS.FieldByName("Num") fmt.Printf("FieldByName("Num") ok:%v field:%+vn", ok, field) // 獲取tag值 fmt.Printf("json tag val:%+vn", field.Tag.Get("json")) // 獲取巢狀結構體的欄位 fmt.Printf("Cnt field:%+vn", typeS.FieldByIndex([]int{2, 0})) }
程式碼輸出如下,
$ go run main.go
NumField:3
Field(0):{Name:Num PkgPath: Type:int Tag:json:"num_json" orm:"column:num_orm" Offset:0 Index:[0] Anonymous:false}
Field(1):{Name:Desc PkgPath: Type:string Tag:json:"desc_json" orm:"column:desc_orm" Offset:8 Index:[1] Anonymous:false}
Field(2):{Name:Child PkgPath: Type:main.secStruct Tag: Offset:24 Index:[2] Anonymous:false}
FieldByName("Num") ok:true field:{Name:Num PkgPath: Type:int Tag:json:"num_json" orm:"column:num_orm" Offset:0 Index:[0] Anonymous:false}
json tag val:num_json
Cnt field:{Name:Cnt PkgPath: Type:[]int64 Tag: Offset:0 Index:[0] Anonymous:false}
通過reflect.ValueOf獲取變數值、值型別,種類為Array, Chan, Map, Slice, 或String可通過Len()獲取長度
func main() { b := int32(1) valueB := reflect.ValueOf(b) fmt.Printf("reflect.TypeOf(b):%v Kind:%vn", valueB, valueB.Kind()) s := "abcdefg" valueS := reflect.ValueOf(s) fmt.Printf("reflect.TypeOf(s):%v Kind:%v Len:%vn", valueS, valueS.Kind(), valueS.Len()) }
程式碼輸出如下,
$ go run main.go
reflect.TypeOf(b):1 Kind:int32
reflect.TypeOf(s):abcdefg Kind:string Len:7
和2.3 結構體成員型別獲取結構體成員型別類似,reflect提供了NumField獲取成員數量,Field通過下標存取成員的值。
func main() { type secStruct struct { Cnt []int64 } type myStruct struct { Num int `json:"num_json" orm:"column:num_orm"` Desc string `json:"desc_json" orm:"column:desc_orm"` Child secStruct } s := myStruct{ Num: 100, Desc: "desc", Child: secStruct{[]int64{1, 2, 3}}, } valueS := reflect.ValueOf(s) // 成員數量 fmt.Printf("NumField:%v n", valueS.NumField()) // 每個成員的值 for i := 0; i < valueS.NumField(); i++ { // 通過下標存取成員 fmt.Printf("value(%v):%+vn", i, valueS.Field(i)) } // 通過名稱存取成員 value := valueS.FieldByName("Num") fmt.Printf("FieldByName("Num") value:%vn", value) // 獲取巢狀結構體的欄位 fmt.Printf("Cnt field:%+vn", valueS.FieldByIndex([]int{2, 0})) }
程式碼輸出如下
$ go run main.go
NumField:3
value(0):100
value(1):desc
value(2):{Cnt:[1 2 3]}
FieldByName("Num") value:100
Cnt field:[1 2 3]
通過func (v Value) Index(i int) Value可以通過下標來存取Array, Slice,或者 String各個元素的值。
func main() { s := []int64{1, 2, 3, 4, 5, 6} valueS := reflect.ValueOf(s) fmt.Printf("ValueOf(s):%v Kind:%v Len:%vn", valueS, valueS.Kind(), valueS.Len()) for i := 0; i < valueS.Len(); i++ { fmt.Printf("valueS.Index(%v):%vn", i, valueS.Index(i)) } }
程式碼輸出如下
$ go run main.go
ValueOf(s):[1 2 3 4 5 6] Kind:slice Len:6
valueS.Index(0):1
valueS.Index(1):2
valueS.Index(2):3
valueS.Index(3):4
valueS.Index(4):5
valueS.Index(5):6
reflect有兩種方法遍歷map
通過迭代器MapIter遍歷map
先獲取map的所有key,再通過key獲取對應的value
func main() { m := map[int]string{ 1: "1", 2: "2", 3: "3", } valueM := reflect.ValueOf(m) // 迭代器存取 iter := valueM.MapRange() for iter.Next() { fmt.Printf("key:%v val:%vn", iter.Key(), iter.Value()) } fmt.Println("------") // 通過key存取 keys := valueM.MapKeys() for i := 0; i < len(keys); i++ { fmt.Printf("key:%v val:%vn", keys[i], valueM.MapIndex(keys[i])) } }
程式碼輸出如下,
$ go run main.go
key:1 val:1
key:2 val:2
key:3 val:3
------
key:3 val:3
key:1 val:1
key:2 val:2
反射的兩個基礎函數定義,
其中,any是interface{}的別名。
interface{}是不包含任何方法簽名的空介面,任何型別都實現了空介面。
A value of interface type can hold any value that implements those methods.
因此,interface{}可以承載任何變數的 (value, concrete type)資訊。
interface{}承載變數的(value, concrete type)資訊,通過反射暴露方法來存取interface{}的值和型別。
可以簡單理解為interface{}的值和資訊傳遞到reflect.Type和 reflect.Value,方便獲取。
可以通過函數func (v Value) Interface() (i any)將反射物件轉換為interface{},
是func ValueOf(i any) Value的反向操作。
func main() { a := int32(10) valueA := reflect.ValueOf(a) fmt.Printf("ValueOf(a):%vn", valueA) fmt.Printf("Interface():%vn", valueA.Interface()) ai, ok := valueA.Interface().(int32) fmt.Printf("ok:%v val:%vn", ok, ai) }
程式碼輸出如下
$ go run main.go
ValueOf(a):10
Interface():10
ok:true val:10
reflect提供func (v Value) CanSet() bool判斷物件值是否修改,通過func (v Value) Set(x Value)修改物件值
func main() { a := int32(10) valueA := reflect.ValueOf(a) fmt.Printf("valueA :%vn", valueA.CanSet()) b := int32(100) valuePtrB := reflect.ValueOf(&b) fmt.Printf("valuePtrB:%v Elem:%vn", valuePtrB.CanSet(), valuePtrB.Elem().CanSet()) valuePtrB.Elem().Set(reflect.ValueOf(int32(200))) fmt.Printf("b:%v Elem:%vn", b, valuePtrB.Elem()) }
程式碼輸出如下
$ go run main.go
valueA :false
valuePtrB:false Elem:true
b:200 Elem:200
到此這篇關於Golang反射獲取變數型別和值的方法詳解的文章就介紹到這了,更多相關Golang反射獲取變數型別 值內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45