揭秘Go語言中的反射機制
基本概念
支持反射的語言可以在程序編譯期將變量的反射信息,如字段名稱、類型信息、結構體信息等整合到可執(zhí)行文件中,并給程序提供接口訪問反射信息,這樣就可以在程序運行期獲取類型的反射信息,并且有能力修改它們。
Go語言提供了 reflect 包來訪問程序的反射信息。
Refelct解析
Refelct包 定義了兩個重要的類型 Type 和 Value,任意接口在反射中都可以理解為 由 reflect.Type 和 reflect.Value 兩部分組成 。簡單來說,go 的接口是由兩部分組成的,一部分是類型信息,另一部分是數據信息
eg
var a=1
var b interface{}=a對于 這個例子,b 的類型信息是 int,數據信息是 1,這兩部分信息都是存儲在 b 里面的。b 的內存結構如下:

而 b實際上是一個空接口,也就是說一個 interface{} 中實際上既包含了變量的類型信息,也包含了類型的數據
refelct.Type ,refelct.Value
如上所說,所有的接口都含有type 和value ,我們可以使用refelct包中的 typeof 和valueof將信息從接口中取出
var a = 1 t := reflect.TypeOf(a) var b = "hello" t1 := reflect.ValueOf(b)
反射定律
三條反射定律 :
- 反射可以將 interface 類型變量轉換成反射對象。
- 反射可以將反射對象還原成 interface 對象。
- 如果要修改反射對象,那么反射對象必須是可設置的(CanSet)。
將 interface 類型變量轉換成反射對象
我們可以通過 reflect.TypeOf 和 reflect.ValueOf 來獲取到一個變量的反射類型和反射值。
var a = 1 typeOfA := reflect.TypeOf(a) valueOfA := reflect.ValueOf(a)
將反射對象還原成 interface 對象。
我們可以通過 reflect.Value.Interface 來獲取到反射對象的 interface 對象,也就是傳遞給 reflect.ValueOf 的那個變量本身。 不過返回值類型是 interface{},所以我們需要進行類型斷言。
i := valueOfA.Interface() fmt.Println(i.(int))
修改反射對象
通過 reflect.Value.CanSet 來判斷一個反射對象是否是可設置的。如果是可設置的,我們就可以通過 reflect.Value.Set 來修改反射對象的值。
var x float64 = 3.4
v := reflect.ValueOf(&x)
fmt.Println("settability of v:", v.CanSet()) // false
fmt.Println("settability of v:", v.Elem().CanSet()) // true那什么情況下一個反射對象是可設置的呢?前提是這個反射對象是一個指針,然后這個指針指向的是一個可設置的變量 .
在上面這個例子中,v.CanSet() 返回的是 false,而 v.Elem().CanSet() 返回的是 true。
在這里,v是一根指針,但是v.Elem()才是v這根指針指向的值。Elem方法是一個解引用的作用。對于這個指針本身,我們修改它是沒有意義的,我們可以設想一下, 如果我們修改了指針變量(也就是修改了指針變量指向的地址),那會發(fā)生什么呢?那樣我們的指針變量就不是指向 x 了, 而是指向了其他的變量,這樣就不符合我們的預期了。所以 v.CanSet() 返回的是 false。
而 v.Elem().CanSet() 返回的是 true。這是因為 v.Elem() 才是 x 本身,通過 v.Elem() 修改 x 的值是沒有問題的

Elem()
refelct.Value中的Elem
reflect.Value 的 Elem 方法的作用是獲取指針指向的值,或者獲取接口的動態(tài)值。
對于指針很好理解,其實作用類似解引用。而對于接口,還是要回到 interface 的結構本身,因為接口里包含了類型和數據本身,所以 Elem 方法就是獲取接口的數據部分(也就是 iface 或 eface 中的 data 字段)。
refelct.Type中的Elem
reflect.Type 的 Elem 方法的作用是獲取數組、chan、map、指針、切片關聯(lián)元素的類型信息,也就是說,對于 reflect.Type 來說, 能調用 Elem 方法的反射對象,必須是數組、chan、map、指針、切片中的一種,其他類型的 reflect.Type 調用 Elem 方法會 panic。
t1 := reflect.TypeOf([3]int{1, 2, 3}) // 數組 [3]int
fmt.Println(t1.String()) // [3]int
fmt.Println(t1.Elem().String()) // int需要注意的是,如果我們要獲取 map 類型 key 的類型信息,需要使用 Key 方法,而不是 Elem 方法。
m := make(map[string]string) t1 := reflect.TypeOf(m) fmt.Println(t1.Key().String()) // string
反射是Go語言中一種強大的編程技術,它允許程序在運行時動態(tài)地檢查和修改對象的結構和行為。通過使用reflect包,我們可以在運行時獲取對象的類型信息、訪問對象的字段和方法、動態(tài)調用方法等。反射在很多場景下都非常有用,比如編寫通用的代碼、實現(xiàn)對象的序列化和反序列化、實現(xiàn)依賴注入等。然而,反射的使用也需要謹慎,因為它會帶來一定的性能損耗。在實際開發(fā)中,我們應該根據具體的需求來判斷是否需要使用反射??偟膩碚f,反射是Go語言中一個非常有用的特性,它為我們提供了更大的靈活性和擴展性。
到此這篇關于揭秘Go語言中的反射機制的文章就介紹到這了,更多相關Go語言反射機制內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
goland安裝1.7版本報錯Unpacked?SDK?is?corrupted解決
這篇文章主要為大家介紹了goland安裝1.7版本報錯Unpacked?SDK?is?corrupted解決,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-11-11
GoFrame?gredis緩存DoVar及Conn連接對象的自動序列化
這篇文章主要為大家介紹了GoFrame?gredis干貨DoVar?Conn連接對象自動序列化詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-06-06
golang 跳出多重循環(huán)的高級break用法說明
這篇文章主要介紹了golang 跳出多重循環(huán)的高級break用法說明,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-12-12
淺析go中Ticker,Timer和Tick的用法與區(qū)別
在go面試的時候,面試官經常會問time包的Ticker,Timer以及Tick的區(qū)別,一般在超時控制的時候用的比較多,今天就跟隨小編一起來詳細學一下這幾個的區(qū)別吧2023-10-10

