Golang反射獲取變數類型和值

来源:https://www.cnblogs.com/amos01/archive/2022/12/03/16945563.html
-Advertisement-
Play Games

如果想直接查看修改部分請跳轉 動手-點擊跳轉 本文基於 ReactiveLoadBalancerClientFilter使用RoundRobinLoadBalancer 灰度發佈 灰度發佈,又稱為金絲雀發佈,是一種新舊版本平滑發佈的方式。在上面可以對同一個API進行兩個版本 的內容,由一部分用戶先行 ...


1. 什麼是反射

反射是程式在運行期間獲取變數的類型和值、或者執行變數的方法的能力。

Golang反射包中有兩對非常重要的函數和類型,兩個函數分別是:

reflect.TypeOf 能獲取類型信息reflect.Type

reflect.ValueOf 能獲取數據的運行時表示reflect.Value

 

3. reflect.Type

Golang是一門靜態類型的語言,反射是建立在類型之上的。

通過reflect.TypeOf() 函數可以獲得任意值的類型信息。

 

3.1 類型Type和種類Kind

諸如int32, slice, map以及通過type關鍵詞自定義的類型。

種類Kind可以理解為類型的具體分類。如int32type MyInt32 int32是兩種不同類型,但都屬於int32這個種類。

使用 reflect.TypeOf()獲取變數類型以及種類。

func main() {
	type MyInt32 int32
	a := MyInt32(1)
	b := int32(1)
	fmt.Printf("reflect.TypeOf(a):%v Kind:%v\n", reflect.TypeOf(a), reflect.TypeOf(a).Kind())
	fmt.Printf("reflect.TypeOf(b):%v Kind:%v\n", reflect.TypeOf(b), reflect.TypeOf(b).Kind())
}

代碼輸出如下,由此可以看出int32type 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
)

 

3.2 引用指向元素的類型

// 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:%v\n", typeA, typeA.Kind())
	fmt.Printf("TypeOf(a).Elem():%v Elem().Kind:%v\n", typeA.Elem(), typeA.Elem().Kind())
	s := []int64{}
	typeS := reflect.TypeOf(s)
	fmt.Printf("TypeOf(s):%v Kind:%v\n", typeS, typeS.Kind())
	fmt.Printf("TypeOf(s).Elem():%v Elem().Kind:%v\n", 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

 

3.3 結構體成員類型

通過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):%+v\n", i, typeS.Field(i))
	}
	// 通過名稱訪問成員
	field, ok := typeS.FieldByName("Num")
	fmt.Printf("FieldByName(\"Num\") ok:%v field:%+v\n", ok, field)
	// 獲取tag值
	fmt.Printf("json tag val:%+v\n", field.Tag.Get("json"))
	// 獲取嵌套結構體的欄位
	fmt.Printf("Cnt field:%+v\n", 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}

 

4. reflect.Value

通過reflect.ValueOf獲取變數值、值類型,種類為Array, Chan, Map, Slice, 或String可通過Len()獲取長度

func main() {
	b := int32(1)
	valueB := reflect.ValueOf(b)
	fmt.Printf("reflect.TypeOf(b):%v Kind:%v\n", valueB, valueB.Kind())
	s := "abcdefg"
	valueS := reflect.ValueOf(s)
	fmt.Printf("reflect.TypeOf(s):%v Kind:%v Len:%v\n", valueS, valueS.Kind(), valueS.Len())
}

代碼輸出如下,

$ go run main.go
reflect.TypeOf(b):1 Kind:int32
reflect.TypeOf(s):abcdefg Kind:string Len:7

 

4.2 結構體的成員的值

3.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):%+v\n", i, valueS.Field(i))
	}
	// 通過名稱訪問成員
	value := valueS.FieldByName("Num")
	fmt.Printf("FieldByName(\"Num\") value:%v\n", value)
	// 獲取嵌套結構體的欄位
	fmt.Printf("Cnt field:%+v\n", 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]

 

4.3 遍歷array、slice

通過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:%v\n", valueS, valueS.Kind(), valueS.Len())
	for i := 0; i < valueS.Len(); i++ {
		fmt.Printf("valueS.Index(%v):%v\n", 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

 

4.4 遍歷map

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:%v\n", iter.Key(), iter.Value())
	}
	fmt.Println("------")
	// 通過key訪問
	keys := valueM.MapKeys()
	for i := 0; i < len(keys); i++ {
		fmt.Printf("key:%v val:%v\n", 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

 

5. 反射的三大定律

反射的兩個基礎函數定義,

  • 獲取類型func TypeOf(i any) Type
  • 獲取值func ValueOf(i any) Value

其中,anyinterface{}的別名。

interface{}是不包含任何方法簽名的空介面,任何類型都實現了空介面。

A value of interface type can hold any value that implements those methods.

因此,interface{}可以承載任何變數的 (value, concrete type)信息。

 

5.1 從interface到反射對象

interface{}承載變數的(value, concrete type)信息,通過反射暴露方法來訪問interface{}的值和類型。

可以簡單理解為interface{}的值和信息傳遞到reflect.Type和 reflect.Value,方便獲取。

 

5.2 從反射對象到interface

可以通過函數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):%v\n", valueA)
	fmt.Printf("Interface():%v\n", valueA.Interface())
	ai, ok := valueA.Interface().(int32)
	fmt.Printf("ok:%v val:%v\n", ok, ai)
}

代碼輸出如下

$ go run main.go
ValueOf(a):10
Interface():10
ok:true val:10

 

5.3 通過反射修改對象,該對象值必須是可修改的

reflect提供func (v Value) CanSet() bool判斷對象值是否修改,通過func (v Value) Set(x Value)修改對象值

func main() {
	a := int32(10)
	valueA := reflect.ValueOf(a)
	fmt.Printf("valueA :%v\n", valueA.CanSet())
	b := int32(100)
	valuePtrB := reflect.ValueOf(&b)
	fmt.Printf("valuePtrB:%v Elem:%v\n", valuePtrB.CanSet(), valuePtrB.Elem().CanSet())
	valuePtrB.Elem().Set(reflect.ValueOf(int32(200)))
	fmt.Printf("b:%v Elem:%v\n", b, valuePtrB.Elem())
}

 代碼輸出如下

$ go run main.go
valueA :false
valuePtrB:false Elem:true
b:200 Elem:200

後續章節再分享通過修改各種類型的值的實操。

 

5. 參考文檔

laws-of-reflection

interface


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • Linux系統環境監測 Linux系統環境主要監測CPU、記憶體、磁碟I/O和網路流量。 1. CPU (1) 查看CPU的負載情況:uptime 可以通過uptime查看系統整體的負載情況。 如果伺服器的CPU為1核心,則1分鐘的系統平均負載 >=3 說明負載過高,如果伺服器的CPU為4核心,則lo ...
  • 1.2 Hadoop簡介 1.2.1 什麼是Hadoop ​ Hadoop 是一個適合大數據的分散式存儲和計算平臺 ​ 如前所述,狹義上說Hadoop就是一個框架平臺,廣義上講Hadoop代表大數據的一個技術生態 圈,包括很多其他軟體框架 ​ Hadoop生態圈技術棧 ​ Hadoop(HDFS + ...
  • 導讀: 今天為大家介紹京東零售大數據的雲原生平臺化實踐,主要包括以下幾大方面內容: 雲原生的定義和理解 雲原生相關技術的演化 京東大數據在雲原生平臺化上的實踐 雲原生應用平臺的發展 分享嘉賓:劉仲偉 京東 架構師 編輯整理:張明宇 廣州某銀行 出品社區:DataFun 01/雲原生的定義和理解 1. ...
  • 一、學習路線 1.HTML5+CSS3 黑馬程式員pink老師前端入門教程,零基礎必看的h5(html5)+css3+移動端前端視頻教程_嗶哩嗶哩_bilibili 2.JavaScript JavaScript基礎語法-dom-bom-js-es6新語法-jQuery-數據可視化echarts黑馬 ...
  • JQuery03 4.jQuery選擇器03 4.4表單選擇器 應用實例 <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>表單選擇器應用實例</title> <script type="text/javasc ...
  • 一、什麼是防禦式編程 防禦性編程是一種細緻、謹慎的編程方法(習慣)。我們在寫代碼時常會有“以防萬一”的心態,把以防萬一有可能出現的情況提前考慮進去,規避以免以防萬一齣現帶來的問題。 應用防禦性編程技術,你可以偵測到可能被忽略的錯誤,防止可能會導致災難性後果的“小毛病”的出現,在時間的運行過程中為你節 ...
  • 散耦合的架構是一種軟體應用程式開發模式,其中多個組件相互連接,但並不嚴重依賴對方。這些組件共同創建了一個總的網路或系統,儘管每個服務都是為執行單一任務而創建的獨立實體。鬆散耦合架構的主要目的是創建一個不會因為單個組件的失敗而失敗的系統。面向服務的架構(SOA)通常由鬆散耦合的架構組成。 鬆散耦合架構 ...
  • 讀本篇文章之前,如果讓你敘述一下 Exception Error Throwable 的區別,你能回答出來麽? 你的反應是不是像下麵一樣呢? 你在寫代碼時會經常 try catch(Exception) 在 log 中會看到 OutOfMemoryError Throwable 似乎不常見,但也大概... ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...