一区二区三区日韩精品-日韩经典一区二区三区-五月激情综合丁香婷婷-欧美精品中文字幕专区

分享

Golang語言學(xué)習(xí)拿捏Go反射示例教程

 新用戶604881da 2021-11-09
目錄

·        1. 反射簡介

·        1.1 反射是什么?

·        1.2 為什么需要反射?

·        2. reflect

·        2.1 基本反射

·        2.2 反射與指針

·        2.3 反射與對象

·        2.4 反射與函數(shù)

·        2.5 反射例子

·        3. 總結(jié)

1. 反射簡介

1.1 反射是什么?

Go語言提供了一種機制在運行時更新和檢查變量的值、調(diào)用變量的方法和變量支持的內(nèi)在操作,但是在編譯時并不知道這些變量的具體類型,這種機制被稱為反射。反射也可以讓我們將類型本身作為第一類的值類型處理。

反射是指在程序運行期對程序本身進行訪問和修改的能力,程序在編譯時變量被轉(zhuǎn)換為內(nèi)存地址,變量名不會被編譯器寫入到可執(zhí)行部分,在運行程序時程序無法獲取自身的信息。

舉個例子
平時我們定義變量都是正射

1

var a int

將變量a定義成一個int類型

現(xiàn)在我并不知道變量a是什么類型,但是我可以通過反射也知曉變量a是什么來歷!是什么類型!

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

type FanOne struct {

    name string

}

func main(){   

    var a int = 1

    var d FanOne

    fmt.Println(reflect.TypeOf(a))            // int

    // 這里就拿到了a的類型!注意是類型!不是類別!雖然這個類型和類別是一樣的

    // 后面會說說類型(Type)和類別(Kind)的區(qū)別

    fmt.Println(reflect.ValueOf(a).Kind())    //int

    //這樣就拿到了a的類別,是通過a的值來判斷類別

    fmt.Println(reflect.TypeOf(d))            //main.FanOne

    //類型是main.FanOne  是在main里面定義的FanOne

    fmt.Println(reflect.ValueOf(d).Kind())    //struct

    //類別是struct

    // 輸出 d 的類型名稱和種類,類型名稱就是 FanOne

    //而 FanOne 屬于一種結(jié)構(gòu)體類別,因此類別為 struct

}

所以這個類別和類型有時候相同,有時候不同。

1.2 為什么需要反射?

在開發(fā)當中,當我們對于某一個函數(shù)進行值的處理的時候,但是為了保證這個函數(shù)能接受更多類型的值,因為go是強類型的語言,雖然interface可以接受所有的數(shù)據(jù)類型,但是在處理數(shù)據(jù)的時候,要對不同類型進行不同的處理的時候就會顯得代碼十分冗余,于是我們可以使用反射來進行對傳入?yún)?shù)的判斷與處理。

詳細見例題

2. reflect

2.1 基本反射

1

2

3

4

reflect.TypeOf()             //獲取變量的類型,返回reflect.Type類型

reflect.ValueOf()            //獲取變量的值,返回reflect.Value類型

reflect.Value.Kind()         //獲取變量的類別,返回一個常量

reflect.Value.Interface()    //轉(zhuǎn)換成interface{}類型

2.2 反射與指針

Go語言程序中對指針獲取反射對象時,可以通過 reflect.Elem() 方法獲取這個指針指向的元素類型,這個獲取過程被稱為取元素,等效于對指針類型變量做了一個*操作

1

reflect.ValueOf(xxx).Elem()

2.3 反射與對象

可以通過reflect.new(xxx)或是reflect.zero(xxx)來進行反射,創(chuàng)建原始類型的對象

1

2

3

func CreatePrimitiveObjects(t reflect.Type) reflect.Value {

    return reflect.Zero(t)

}

也可以使用

1

reflect.New()

來進行創(chuàng)建原始對象。

2.4 反射與函數(shù)

如果反射值對象(reflect.Value)中值的類型為函數(shù)時,可以通過reflect.Value調(diào)用該函數(shù)。使用反射調(diào)用函數(shù)時,需要將參數(shù)使用反射值對象的切片[]reflect.Value構(gòu)造后傳入Call()方法中,調(diào)用完成時,函數(shù)的返回值通過[]reflect.Value返回。
在反射中 函數(shù) 和 方法 的類型(Type)都是reflect.Func,如果要調(diào)用函數(shù)的話,可以通過 Value 的 Call() 方法,例如:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

package main

import (

    "fmt"

    "reflect"

)

func FanOne() string {

    return "一鍵三連"

}

func FanOneWoW(a string) string {

    return fmt.Sprintf("%s要給FanOne一鍵三連噢~",a)

}

func main() {

    FanOneNotArgs := reflect.ValueOf(FanOne).Call([]reflect.Value{})         //無參數(shù)

    FanOneHaveArgs := reflect.ValueOf(FanOneWoW).Call([]reflect.Value{reflect.ValueOf("我")})  //有參數(shù)

    fmt.Println(FanOneNotArgs[0])

    fmt.Println(FanOneHaveArgs[0])

}

2.5 反射例子

填寫fn函數(shù)使得輸出為

要求不使用任何的switch 或是 if 或是其他選擇語句。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

func fn(callback interface{}, bytes []byte) {

        //coding

}

type aaa struct {

    Name string `json:"name"`

    Age  int    `json:"age"`

}

func Test(t *testing.T) {

        fn(func(a []*aaa) string {

        aaas := a

        for i, item := range aaas {

            fmt.Println(i, item)

        }

        fmt.Println("12312312, ", aaas)

        return "xxxx"

    }, []byte("[{\"name\":\"111\",\"age\":1}, {\"name\":\"gsjk\",\"age\":2}]"))

    fn(func(a []aaa) string {

        aaas := a

        for i, item := range aaas {

            fmt.Println(i, item)

        }

        fmt.Println("12312312, ", aaas[0])

        return "xxxx"

    }, []byte("[{\"name\":\"111\",\"age\":1}, {\"name\":\"gsjk\",\"age\":2}]"))

    fn(func(a *aaa) string {

        fmt.Println("12312312, ", a)

        aaas := a

        fmt.Println("12312312, ", aaas)

        return "xxxx"

    }, []byte("{\"name\":\"gsjk\",\"age\":2}"))

    fn(func(a string) string {

        fmt.Println("12312312, ", a)

        aaas := a

        fmt.Println("12312312, ", aaas)

        return "xxxx"

    }, []byte("\"sss\""))

    fn(func(a int) string {

        fmt.Println("-----------, ", a)

        aaas := a

        fmt.Println("-----------, ", aaas)

        return "xxxx"

    }, []byte("123"))

}

(1)首先是test的知識:
名稱一定要有_test,不然好像會報錯,我就是這樣。

1

2

go test xxx_test.go

go test -v xxx_test.go

(2)其次是了解這個fn()里面的匿名函數(shù)
單獨拿出來

1

2

3

4

5

6

7

8

func(a []*aaa) string {

        aaas := a

        for i, item := range aaas {

            fmt.Println(i, item)

        }

        fmt.Println("12312312, ", aaas)

        return "xxxx"

    }, []byte("[{\"name\":\"111\",\"age\":1}, {\"name\":\"gsjk\",\"age\":2}]"))

可以看到這是一個*aaa類型的數(shù)組。那么我們?nèi)蝿?wù)就是反射出fn這個函數(shù)里面的匿名函數(shù),然后調(diào)用反射出來的這個匿名函數(shù),并將參數(shù)傳入其中。

以下都是用第一個作為例子

(3)那么我們先ValueOf和TypeOf這個interface{},然后再看這個匿名函數(shù)各種的值

1

2

3

4

func fn(callback interface{}, bytes []byte) {

    v := reflect.ValueOf(callback)  //0xbaff40

    t := reflect.TypeOf(callback)  //func([]*main.aaa) string

}

我們可以看到入?yún)⒌暮瘮?shù)的Type是func([]*main.aaa) string 所以我們可以用

1

paramsValue := t.In(0)  //[]*main.aaa

拿到匿名函數(shù)的傳入?yún)?shù)

(4)重點??!
我們拿到的這個
paramsValue只是[]*main.aaa名稱,這個值是reflect.type 類型的??!、
我們要的是
[]*main.aaa這個類型,而不是要這個名稱!
所以我們要創(chuàng)建這個類型的對象,然后轉(zhuǎn)成相應(yīng)的類型

1

2

3

4

val := reflect.New(paramsValue)

newT := val.Interface()

fmt.Printf("valValue:%v , valType: %T \n",val,val)    //valValue:&[] , valType: reflect.Value

fmt.Printf("newTValue:%v , newTType: %T \n",newT,newT)//newTValue:&[] , newTType: *[]*main.aaa

我們要創(chuàng)建這樣一個類別的對象,雖然go并不是面向?qū)ο蟮木幊?,但是這里可以這樣理解。

為什么要這個類型呢?

因為后面把bytes切片反序列化成這個類型的變量,傳入這個匿名函數(shù)中!

1

2

3

if v.IsValid() {                                //function valid or not

    _ = json.Unmarshal(bytes, newT)               //byte to json

}

那么問題又來了,傳入的值的類型是[]*main.aaa 但是我們拿到了*[]*main.aaa這個類型,很明顯是不對的。

1

2

fmt.Printf("調(diào)用 callback 結(jié)束 callback ret = %s \n", v.Call([]reflect.Value{reflect.ValueOf(newT)}))

fmt.Printf("*************************\n")

報錯了!類型不對!

那么我們就要進行去*操作。在反射中,并不是直接加*去除!下面這樣在反射中是不行的。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

package main

import (

  "fmt"

  "reflect"

)

func main(){

  var a int = 1

  var b *int = &a

  var c **int = &b

  fmt.Println(a, *b, c)

  fmt.Println(reflect.TypeOf(a))

  fmt.Println(reflect.TypeOf(*b))

  fmt.Println(reflect.TypeOf(b))

}

那么我們可以用reflect.Elem() 將這個去除*

1

2

fmt.Printf("調(diào)用 callback 結(jié)束 callback ret = %s \n", v.Call([]reflect.Value{reflect.ValueOf(newT).Elem()}))

fmt.Printf("*************************\n")

大功告成了!

3. 總結(jié)

以前我是很少使用反射的,基本在項目中就沒用過,但是暑期實習(xí)的時候,第一個任務(wù)就是寫反射接口,那么就瘋狂補這方面的知識,反射對于我來說,確實有點難理解,花了我兩天時間才做出來。

原來我的想法是用if判斷類型的,或是斷言然后用switch判斷類型的,但是這樣實用性不高,換了一個名稱的話就要改代碼了。比如現(xiàn)在是結(jié)構(gòu)體aaa,換成bbb就不管用了。

現(xiàn)在這種的話,直接將這個類型的反射成一個對象,然后再對這個對象進行賦值操作,就更加靈活!

學(xué)到了!

實習(xí)很痛苦!但是學(xué)到了很多新知識!還有好多大佬帶!還有工資拿!也舒服!

以上就是Golang語言學(xué)習(xí)拿捏Go反射示例教程的詳細內(nèi)容,更多關(guān)于Go反射教程的資料請關(guān)注腳本之家其它相關(guān)文章!

    本站是提供個人知識管理的網(wǎng)絡(luò)存儲空間,所有內(nèi)容均由用戶發(fā)布,不代表本站觀點。請注意甄別內(nèi)容中的聯(lián)系方式、誘導(dǎo)購買等信息,謹防詐騙。如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,請點擊一鍵舉報。
    轉(zhuǎn)藏 分享 獻花(0

    0條評論

    發(fā)表

    請遵守用戶 評論公約

    類似文章 更多

    欧美日韩国产一级91| 国产欧美日产久久婷婷| 69精品一区二区蜜桃视频| 黄色在线免费高清观看| 国产又色又爽又黄又免费| 国产乱淫av一区二区三区| 色无极东京热男人的天堂| 亚洲中文字幕三区四区| 国产伦精品一一区二区三区高清版| 亚洲午夜福利不卡片在线| 欧美国产在线观看精品| 午夜福利大片亚洲一区| 成人午夜免费观看视频| 白丝美女被插入视频在线观看| 国产男女激情在线视频| 欧美野外在线刺激在线观看| 久久亚洲精品成人国产| 午夜福利直播在线视频| 国产精品丝袜一二三区| 欧美激情一区=区三区| 午夜福利视频日本一区| 激情中文字幕在线观看| 日韩aa一区二区三区| 日韩在线视频精品中文字幕| 粉嫩国产美女国产av| 又大又紧又硬又湿又爽又猛| 午夜视频免费观看成人| 不卡一区二区在线视频| 国产一区二区精品高清免费| 最好看的人妻中文字幕| 男女午夜福利院在线观看| 日韩综合国产欧美一区| 少妇激情在线免费观看| 中文字幕亚洲在线一区| 日韩欧美在线看一卡一卡| 麻豆精品在线一区二区三区| 精品视频一区二区三区不卡| 青青操视频在线观看国产| 欧美国产精品区一区二区三区| 国产欧美亚洲精品自拍| 91欧美一区二区三区成人|