Go: 反射

反射是Go中的高级语法,本文将以尽可能用简单的方式来介绍它。
本文包括以下部分:

  • 什么是反射?
  • 如何检查一个变量及其类型?
  • 反射包
  • 完整代码
  • 应该用反射吗?
    现在我们一个一个地讨论这些部分。

什么是反射?

反射是程序在运行时能够检查变量、变量值和变量类型的能力。现在你可能不理解什么意思,但是不要紧。在读完本文,您将对反射有一个清晰的理解,所以请继续跟随我。

检查一个变量及其类型需要什么?

在学习反射时每个人碰到第一个问题就是程序运行时为何需要检查变量类型,程序中的每一个变量都是由我们定义的,并且我们在编译时就知道它的类型。这在大多数情况下是正确的,但并不总是如此。
让我们写一个简单的程序,来解释下。

package main

import (  
    "fmt"
)

func main() {  
    i := 10
    fmt.Printf("%d %T", i, i)
}

在上面的代码中,变量i的类型在编译的时候就知道了,并且在代码中打印。没什么特别的。

现在我们来看为何在运行时需要知道变量的类型。假设我们想要编写一个简单的函数,它接受一个结构体作为参数,并使用它创建一个SQL查询语句。看以下代码:

package main

import (  
    "fmt"
)

type order struct {  
    ordId      int
    customerId int
}

func main() {  
    o := order{
        ordId:      1234,
        customerId: 567,
    }
    fmt.Println(o)
}

我们需要编写一个函数,它将接受上面程序中的结构体o作为参数,并返回以下SQL插入查询:

insert into order values(1234, 567)  

实现这个函数很简单:

package main

import (  
    "fmt"
)

type order struct {  
    ordId      int
    customerId int
}

func createQuery(o order) string {  
    i := fmt.Sprintf("insert into order values(%d, %d)", o.ordId, o.customerId)
    return i
}

func main() {  
    o := order{
        ordId:      1234,
        customerId: 567,
    }
    fmt.Println(createQuery(o))
}

createQuery函数使用结构体的ordId和customerId字段生成sql查询语句。程序输出为:

insert into order values(1234, 567)  

现在将createQuery函数再升级一下。如果我们想要泛化createQuery函数,并让它在任何结构体上工作,该怎么办?我解释一下我所说的是什么意思。

package main

type order struct {  
    ordId      int
    customerId int
}

type employee struct {  
    name string
    id int
    address string
    salary int
    country string
}

func createQuery(q interface{}) string {  
}

func main() {

}

我们的目标是实现createQuery函数能够接收任何结构体为参数,并根据结构体字段来创建查询语句。
假设,传入以下结构体:

o := order {  
    ordId: 1234,
    customerId: 567
}

createQuery函数应该返回:

insert into order values (1234, 567)  

类似地,传入:

e := employee {
        name: "Naveen",
        id: 565,
        address: "Science Park Road, Singapore",
        salary: 90000,
        country: "Singapore",
    }

应该返回:

insert into employee values("Naveen", 565, "Science Park Road, Singapore", 90000, "Singapore")  

因为createQuery函数对任何结构体都生效,它将interface{}类型作为参数类型。为了简单,我们只处理结构体包含string类型和int类型,但可以扩展到任意类型。

要实现createQuery函数对任意结构体类型都生效。唯一的办法就是在运行时检查结构体参数的类型,并找到各字段后,创建查询语句。这就是反射的作用。下一步,我们将学习如何使用reflect包来实现。

reflect包

reflect包实现了Go运行时反射。反射包能够识别interface{}变量的底层类型和值。这也是我们需要的。createQuery函数接收interface{}类型参数,需要根据其具体类型和值来创建查询语句。也是reflect包实现的功能。

在编写通用createQuery函数之前,我们需要了解反射包中的一些类型和方法。让我们一个一个来看看。

reflect.Type和reflect.Value

interface{}变量具体类型由reflect.Type表示,而reflect.Value表示其具体值。reflect.TypeOf()函数和reflect.ValueOf()函数分别返回reflect.Type和reflect.Value。这两种类型是实现createQuery函数的基础。让我们编写一个简单的示例来理解这两种类型。

package main

import (  
    "fmt"
    "reflect"
)

type order struct {  
    ordId      int
    customerId int
}

func createQuery(q interface{}) {  
    t := reflect.TypeOf(q)
    v := reflect.ValueOf(q)
    fmt.Println("Type ", t)
    fmt.Println("Value ", v)


}
func main() {  
    o := order{
        ordId:      456,
        customerId: 56,
    }
    createQuery(o)

}

上面的代码,createQuery函数接收interface{}类型参数。reflect.TypeOf接收interface{}类型并返回reflect.Type,包含interface{}参数的具体类型。类似地,reflect.ValueOf函数也是接收interface{}类型返回reflect.Value,包含所传递的interface{}参数的具体值。
代码输出为:

Type  main.order
Value  {456 56}

从输出中,我们可以看到程序输出了接口的具体类型和值。

reflect.Kind

在反射包中还有一种更重要的类型,称为Kind。
反射包中的Kind和Type类型似乎很相似,但他们有一个区别,从下面的程序中可以清楚地看到。

package main

import (  
    "fmt"
    "reflect"
)

type order struct {  
    ordId      int
    customerId int
}

func createQuery(q interface{}) {  
    t := reflect.TypeOf(q)
    k := t.Kind()
    fmt.Println("Type ", t)
    fmt.Println("Kind ", k)


}
func main() {  
    o := order{
        ordId:      456,
        customerId: 56,
    }
    createQuery(o)

}

程序输出为:

Type  main.order  
Kind  struct 

我想你现在应该明白两者之间的区别了。Type表示interface{}的实际类型,在本例中为main.Order,Kind表示特定类型。在这个例子中它是一个struct。

NumField()和Field()方法

NumField()方法返回结构中字段的数量,Field(i int)方法返回字段I的reflect.Value。

package main

import (  
    "fmt"
    "reflect"
)

type order struct {  
    ordId      int
    customerId int
}

func createQuery(q interface{}) {  
    if reflect.ValueOf(q).Kind() == reflect.Struct {
        v := reflect.ValueOf(q)
        fmt.Println("Number of fields", v.NumField())
        for i := 0; i < v.NumField(); i++ {
            fmt.Printf("Field:%d type:%T value:%v\n", i, v.Field(i), v.Field(i))
        }
    }

}
func main() {  
    o := order{
        ordId:      456,
        customerId: 56,
    }
    createQuery(o)
}

以上代码,我们先检查参数q的Kind是否为struct,因为NumField方法只对struct类型有效。 其余部分很好理解。这个程序输出:

Number of fields 2  
Field:0 type:reflect.Value value:456  
Field:1 type:reflect.Value value:56  

Int()和String()方法

Int和String方法将reflect.Value值提取出来分别是int64和string类型。

package main

import (  
    "fmt"
    "reflect"
)

func main() {  
    a := 56
    x := reflect.ValueOf(a).Int()
    fmt.Printf("type:%T value:%v\n", x, x)
    b := "Naveen"
    y := reflect.ValueOf(b).String()
    fmt.Printf("type:%T value:%v\n", y, y)

}

代码输出:

type:int64 value:56  
type:string value:Naveen  

完整代码

现在我们已经有了足够的知识来完成createQuery函数,让我们继续完成它。

package main

import (  
    "fmt"
    "reflect"
)

type order struct {  
    ordId      int
    customerId int
}

type employee struct {  
    name    string
    id      int
    address string
    salary  int
    country string
}

func createQuery(q interface{}) {  
    if reflect.ValueOf(q).Kind() == reflect.Struct {
        t := reflect.TypeOf(q).Name()
        query := fmt.Sprintf("insert into %s values(", t)
        v := reflect.ValueOf(q)
        for i := 0; i < v.NumField(); i++ {
            switch v.Field(i).Kind() {
            case reflect.Int:
                if i == 0 {
                    query = fmt.Sprintf("%s%d", query, v.Field(i).Int())
                } else {
                    query = fmt.Sprintf("%s, %d", query, v.Field(i).Int())
                }
            case reflect.String:
                if i == 0 {
                    query = fmt.Sprintf("%s\"%s\"", query, v.Field(i).String())
                } else {
                    query = fmt.Sprintf("%s, \"%s\"", query, v.Field(i).String())
                }
            default:
                fmt.Println("Unsupported type")
                return
            }
        }
        query = fmt.Sprintf("%s)", query)
        fmt.Println(query)
        return

    }
    fmt.Println("unsupported type")
}

func main() {  
    o := order{
        ordId:      456,
        customerId: 56,
    }
    createQuery(o)

    e := employee{
        name:    "Naveen",
        id:      565,
        address: "Coimbatore",
        salary:  90000,
        country: "India",
    }
    createQuery(e)
    i := 90
    createQuery(i)

}

以上代码,我们先检查传入的参数是否为struct类型。然后使用reflect.Type的Name()方法获取结构体名称。并使用switch—case来判断各字段的具体类型,分别处理。

我们还添加了检查,防止将不支持的类型传递给createQuery函数时程序崩溃。程序的其余部分很好理解。我建议在适当的位置添加日志,并检查它们的输出,以便更好地理解这个程序。
程序输出为:

insert into order values(456, 56)  
insert into employee values("Naveen", 565, "Coimbatore", 90000, "India")  
unsupported type  

将字段名添加到输出查询中,留给读者作为练习。请尝试更改程序打印查询的格式,

应该使用反射吗?

在展示了反射的实际应用之后,现在真正的问题来了。应该使用反射吗?我想引用Rob Pike的话来回答这个问题。

Clear is better than clever. Reflection is never clear.

反射在Go中是一个非常强大和先进的概念,应该谨慎使用。使用反射编写清晰和可维护的代码是非常困难的。在任何可能的情况下都应该避免使用,只有在绝对必要的时候才应该使用。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,951评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,606评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,601评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,478评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,565评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,587评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,590评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,337评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,785评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,096评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,273评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,935评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,578评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,199评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,440评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,163评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,133评论 2 352

推荐阅读更多精彩内容