Go語言反射獲取類型屬性和方法示例

reflect.StructField 和 reflect.Method

如果變量是一個結構體,我們還可以通過結構體域類型對象 reflect.StructField 來獲取結構體下字段的類型屬性。Type 接口下提供瞭不少用於獲取字段結構體域類型對象的方法,我們主要介紹以下幾個接口:

	// 獲取一個結構體內的字段數量
	NumField() int
	// 根據 index 獲取結構體內的成員字段類型對象
	Field(i int) StructField
	// 根據字段名獲取結構體內的成員字段類型對象
	FieldByName(name string) (StructField, bool)

通過以上的 3 個方法,我們可以輕易地拿到一個結構體變量內的所有成員字段的類型對象 reflect.StructField。通過 reflect.StructField,我們可以知道成員字段所屬的類型和種類,其內主要由以下的屬性:

type StructField struct {
	// 成員字段的名稱
	Name string
	// 成員字段 Type
	Type      Type
	// Tag
	Tag       StructTag
	// 字節偏移
	Offset    uintptr
	// 成員字段的 index
	Index     []int
	// 成員字段是否公開
	Anonymous bool
}

StructField

StructField 中提供瞭 Type 用於獲取字段的的類型信息,而 StructTag 一般用來描述結構體成員字段的額外信息,比如在 JSON 進行序列化和對象映射時會被使用。StructTag 一般由一個或者多個鍵值對組成,一個簡單的例子如下:

ID string `json:"id"`

鍵與值使用 : 分隔,值用 "" 括起來, 鍵值對之間使用空格分隔。上面例子中說明 ID 字段在 JSON 序列化時會被變成 id 。

遍歷 Hero 結構體

接下來,我們通過遍歷 Hero 結構體,獲取其內字段的類型並輸出,代碼如下所示:

func main()  {
	typeOfHero := reflect.TypeOf(Hero{})
	// 通過 #NumField 獲取結構體字段的數量
	for i := 0 ; i < typeOfHero.NumField(); i++{
		fmt.Printf("field' name is %s, type is %s, kind is %s\n", typeOfHero.Field(i).Name, typeOfHero.Field(i).Type, typeOfHero.Field(i).Type.Kind())
	}
	// 獲取名稱為 Name 的成員字段類型對象
	nameField, _ := typeOfHero.FieldByName("Name")
	fmt.Printf("field' name is %s, type is %s, kind is %s\n", nameField.Name, nameField.Type, nameField.Type.Kind())
}

預期的結果如下所示:

field' name is Name, type is string, kind is string
field' name is Age, type is int, kind is int
field' name is Speed, type is int, kind is int
field' name is Name, type is string, kind is string

上述代碼中先使用 Type#NumField 獲取 Hero 結構體中字段的數量,再通過 typeOfHero#Field 根據 index 獲取每個字段域類型對象並打印它們的類型信息。代碼最後還演示如何通過 typeOfHero#FieldByName 獲取瞭字段名為 Name 的字段域類型對象。

Method

除瞭獲取結構體下的字段域類型對象,Type 還提供方法獲取接口下方法的方法類型對象 Method,接口方法描述如下:

	// 根據 index 查找方法
	Method(int) Method
	// 根據方法名查找方法
	MethodByName(string) (Method, bool)
	// 獲取類型中公開的方法數量
	NumMethod() int

獲取到的方法類型描述對象 Method 描述瞭方法的基本信息,包括方法名,方法類型等,代碼如下所示:

type Method struct {
	// 方法名
	Name    string
	// 方法類型
	Type  Type
	// 反射對象,可用於調用方法
	Func  Value
	// 方法的index
	Index int
}

在 Method 中 Func 字段是一個反射值對象,可用於進行方法的調用。如果 Method 是來自於接口類型反射得到的 Type ,那麼 Func 傳遞的第一個參數需要為實現方法的接收器,這部分區別我們將在 Value 中進行具體的介紹。

我們可以通過 Type 中提供的方法獲取接口 Person 中方法的方法類型對象,代碼如下所示:

func main()  {
	// 聲明一個 Person 接口,並用 Hero 作為接收器
		var person Person = &Hero{}
	// 獲取接口Person的類型對象
	typeOfPerson := reflect.TypeOf(person)
	// 打印Person的方法類型和名稱
	for i := 0 ; i < typeOfPerson.NumMethod(); i++{
		fmt.Printf("method is %s, type is %s, kind is %s.\n", typeOfPerson.Method(i).Name, typeOfPerson.Method(i).Type, typeOfPerson.Method(i).Type.Kind())
	}
	method, _ := typeOfPerson.MethodByName("Run")
	fmt.Printf("method is %s, type is %s, kind is %s.\n", method.Name, method.Type, method.Type.Kind())
	}

預期的輸出結果如下所示:

method is Run, type is func(*main.Hero), kind is func
method is SayHello, type is func(*main.Hero, string), kind is func
method is Run, type is func(*main.Hero) string, kind is func.

除瞭通過 typeOfPerson#Method 根據 index 獲取方法類型對象,還可以使用 typeOfPerson#MethodByName 根據方法名查找對應的方法類型對象。從輸出結果可以看出,方法的種類均為 func,而類型則為方法的聲明。

小結

本文主要介紹瞭 Go 語言的反射基礎 reflect.StructField 和 reflect.Method。通過反射,我們可以拿到類型信息和定義的方法等,Go 的反射實現瞭反射的大多數功能,獲取類型信息需要配合使用標準庫中的詞法、語法解析器和抽象語法數對源碼進行掃描。下一篇文章將會繼續介紹 Go 語言的反射 reflect.Value 反射值對象相關內容。

以上就是Go語言反射獲取類型屬性和方法示例的詳細內容,更多關於Go反射獲取類型屬性方法的資料請關註WalkonNet其它相關文章!

推薦閱讀: