在 Go語言中通過調(diào)用 reflect.TypeOf 函數(shù),我們可以從一個(gè)任何非接口類型的值創(chuàng)建一個(gè) reflect.Type 值。reflect.Type 值表示著此非接口值的類型。通過此值,我們可以得到很多此非接口類型的信息。當(dāng)然,我們也可以將一個(gè)接口值傳遞給一個(gè) reflect.TypeOf 函數(shù)調(diào)用,但是此調(diào)用將返回一個(gè)表示著此接口值的動(dòng)態(tài)類型的 reflect.Type 值。
實(shí)際上,reflect.TypeOf 函數(shù)的唯一參數(shù)的類型為 interface{},reflect.TypeOf 函數(shù)將總是返回一個(gè)表示著此唯一接口參數(shù)值的動(dòng)態(tài)類型的 reflect.Type 值。
那如何得到一個(gè)表示著某個(gè)接口類型的 reflect.Type 值呢?我們必須通過下面將要介紹的一些間接途徑來達(dá)到這一目的。
類型 reflect.Type 為一個(gè)接口類型,它指定了若干方法(https://golang.google.cn/pkg/reflect/#Type)。 通過這些方法,我們能夠觀察到一個(gè) reflect.Type 值所表示的 Go類型的各種信息。這些方法中的有的適用于所有種類(https://golang.google.cn/pkg/reflect/#Kind)的類型,有的只適用于一種或幾種類型。通過不合適的 reflect.Type 屬主值調(diào)用某個(gè)方法將在運(yùn)行時(shí)產(chǎn)生一個(gè)恐慌。
使用 reflect.TypeOf() 函數(shù)可以獲得任意值的類型對象(reflect.Type),程序通過類型對象可以訪問任意值的類型信息。下面通過例子來理解獲取類型對象的過程:
package main
import (
"fmt"
"reflect"
)
func main() {
var a int
typeOfA := reflect.TypeOf(a)
fmt.Println(typeOfA.Name(), typeOfA.Kind())
}
代碼輸出如下:
int int
代碼說明如下:
- 第 10 行,定義一個(gè) int 類型的變量。
- 第 12 行,通過 reflect.TypeOf() 取得變量 a 的類型對象 typeOfA,類型為 reflect.Type()。
- 第 14 行中,通過 typeOfA 類型對象的成員函數(shù),可以分別獲取到 typeOfA 變量的類型名為 int,種類(Kind)為 int。
理解反射的類型(Type)與種類(Kind)
在使用反射時(shí),需要首先理解類型(Type)和種類(Kind)的區(qū)別。編程中,使用最多的是類型,但在反射中,當(dāng)需要區(qū)分一個(gè)大品種的類型時(shí),就會(huì)用到種類(Kind)。例如,需要統(tǒng)一判斷類型中的指針時(shí),使用種類(Kind)信息就較為方便。
1) 反射種類(Kind)的定義
Go 程序中的類型(Type)指的是系統(tǒng)原生數(shù)據(jù)類型,如 int、string、bool、float32 等類型,以及使用 type 關(guān)鍵字定義的類型,這些類型的名稱就是其類型本身的名稱。例如使用 type A struct{} 定義結(jié)構(gòu)體時(shí),A 就是 struct{} 的類型。
種類(Kind)指的是對象歸屬的品種,在 reflect 包中有如下定義:
type Kind uint
const (
Invalid Kind = iota // 非法類型
Bool // 布爾型
Int // 有符號整型
Int8 // 有符號8位整型
Int16 // 有符號16位整型
Int32 // 有符號32位整型
Int64 // 有符號64位整型
Uint // 無符號整型
Uint8 // 無符號8位整型
Uint16 // 無符號16位整型
Uint32 // 無符號32位整型
Uint64 // 無符號64位整型
Uintptr // 指針
Float32 // 單精度浮點(diǎn)數(shù)
Float64 // 雙精度浮點(diǎn)數(shù)
Complex64 // 64位復(fù)數(shù)類型
Complex128 // 128位復(fù)數(shù)類型
Array // 數(shù)組
Chan // 通道
Func // 函數(shù)
Interface // 接口
Map // 映射
Ptr // 指針
Slice // 切片
String // 字符串
Struct // 結(jié)構(gòu)體
UnsafePointer // 底層指針
)
Map、Slice、Chan 屬于引用類型,使用起來類似于指針,但是在種類常量定義中仍然屬于獨(dú)立的種類,不屬于 Ptr。
type A struct{} 定義的結(jié)構(gòu)體屬于 Struct 種類,*A 屬于 Ptr。
2) 從類型對象中獲取類型名稱和種類的例子
Go語言中的類型名稱對應(yīng)的反射獲取方法是 reflect.Type 中的 Name() 方法,返回表示類型名稱的字符串。
類型歸屬的種類(Kind)使用的是 reflect.Type 中的 Kind() 方法,返回 reflect.Kind 類型的常量。
下面的代碼中會(huì)對常量和結(jié)構(gòu)體進(jìn)行類型信息獲取。
package main
import (
"fmt"
"reflect"
)
// 定義一個(gè)Enum類型
type Enum int
const (
Zero Enum = 0
)
func main() {
// 聲明一個(gè)空結(jié)構(gòu)體
type cat struct {
}
// 獲取結(jié)構(gòu)體實(shí)例的反射類型對象
typeOfCat := reflect.TypeOf(cat{})
// 顯示反射類型對象的名稱和種類
fmt.Println(typeOfCat.Name(), typeOfCat.Kind())
// 獲取Zero常量的反射類型對象
typeOfA := reflect.TypeOf(Zero)
// 顯示反射類型對象的名稱和種類
fmt.Println(typeOfA.Name(), typeOfA.Kind())
}
代碼輸出如下:
cat struct
Enum int
代碼說明如下:
- 第 18 行,聲明結(jié)構(gòu)體類型 cat。
- 第 22 行,將 cat 實(shí)例化,并且使用 reflect.TypeOf() 獲取被實(shí)例化后的 cat 的反射類型對象。
- 第 25 行,輸出cat的類型名稱和種類,類型名稱就是 cat,而 cat 屬于一種結(jié)構(gòu)體種類,因此種類為 struct。
- 第 28 行,Zero 是一個(gè) Enum 類型的常量。這個(gè) Enum 類型在第 9 行聲明,第 12 行聲明了常量。如沒有常量也不能創(chuàng)建實(shí)例,通過 reflect.TypeOf() 直接獲取反射類型對象。
- 第 31 行,輸出 Zero 對應(yīng)的類型對象的類型名和種類。
到此這篇關(guān)于Go語言reflect.TypeOf()和reflect.Type通過反射獲取類型信息 的文章就介紹到這了,更多相關(guān)Go 反射獲取類型信息 內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
您可能感興趣的文章:- go語言靜態(tài)庫的編譯和使用方法
- Go語言-為什么返回值為接口類型,卻返回結(jié)構(gòu)體
- Go語言操作數(shù)據(jù)庫及其常規(guī)操作的示例代碼
- Go語言利用ssh連接服務(wù)器的方法步驟
- 詳解用Go語言實(shí)現(xiàn)工廠模式(Golang經(jīng)典編程案例)
- go語言實(shí)現(xiàn)簡易比特幣系統(tǒng)之交易簽名及校驗(yàn)功能
- Go語言編譯程序從后臺(tái)運(yùn)行,不出現(xiàn)dos窗口的操作