1 概述

类似于 Java,Go 语言也支持反射。支持反射的语言可以在运行时对程序进行访问和修改。反射的原理是在程序编译期将反射信息(如类型信息、结构体信息等)整合到程序中,并给提供给程序访问反射信息的操作接口,这样在程序运行期间就可以获取该反射信息,甚至支持修改操作。

Go 语言使用 reflect 包支持反射。

本文介绍与类型结构相关的反射操作。

2 获取类型

使用 reflect.TypeOf() 函数可以获得任意值的类型反射对象。演示为:

type Stu struct {
}
var v *Stu
typeV := reflect.TypeOf(v)
fmt.Println(typeV)
// Stu

其中,typeV是 reflect.Type 类型的实例。

3 获取基础类型(类别)

基础类型,也称之为类别。例如 type Stu struct,从类型上看是 Stu 类型,如果从基础类型(类别)的角度去看,就是 struct。当需要区分一个大类类别时,就会用到基础类型的概念。可以通过 typeV.Kind() 方法获取对应的基础类型。演示为:

type Stu struct {
}
var v Stu
typeV := reflect.TypeOf(v)
// 同时输出类型名和基础类型
fmt.Println(typeV.Name(), typeV.Kind())
// Stu struct

Go 语言的 reflect 包定义了如下的基础类型:

来自文件:src/reflect/type.go
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 // 单精度浮点数
Float64 // 双精度浮点数
Complex64 // 64位复数类型
Complex128 // 128位复数类型
Array // 数组
Chan // 通道
Func // 函数
Interface // 接口
Map // 映射
Ptr // 指针
Slice // 切片
String // 字符串
Struct // 结构体
UnsafePointer // 底层指针
)

可见指的是原生类型,而不是自定义类型。

4 指针引用的元素类型

可以使用指针类型的反射得到其指向的元素的具体类型,使用 Elem() Type 来实现,演示为:

type Stu struct {
}
var v *Stu
typeV := reflect.TypeOf(v)
fmt.Println(typeV)
// *main.Stu
fmt.Println(typeV.Kind())
// 基础类型为 ptr 指针
// ptr
fmt.Println(typeV.Elem())
// 指向的元素类型为 main.Stu
// main.Stu
fmt.Println(typeV.Elem().Kind())
// main.Stu的基础类型为 struct
// struct

.Elem() 方法会得到 reflect.Type 类型的返回值,因此可以继续调用 .Kind() 得到基础类型。

5 结构体信息

若反射的类型为结构体,可以获取其成员信息。涉及几个方法:

  • NumField() int,字段数量
  • Field(i int) StructField,通过索引确定获取字段的反射
  • NumMethod() int,方法数量
  • Method(int) Method,通过索引获取方法的反射

演示为:

type Stu struct {
Name string
Sn string
} func (this *Stu) SetName() {
}
func (this *Stu) SetSn() {
} func main() {
v := Stu{}
typeV := reflect.TypeOf(v)
fmt.Println(typeV.NumField())
for i, c := 0, typeV.NumField(); i < c; i++ {
fmt.Println(typeV.Field(i))
}
vp := &v // ? 为什么必须要是引用呢 ?
typeVP := reflect.TypeOf(vp)
fmt.Println(typeVP.NumMethod())
for i, c := 0, typeV.NumMethod(); i < c; i++ {
fmt.Println(typeVP.Method(i))
}
}
// 以下为输出结果
2
{Name string 0 [0] false}
{Sn string 16 [1] false}
2
{SetName func(*main.Stu) <func(*main.Stu) Value> 0}
{SetSn func(*main.Stu) <func(*main.Stu) Value> 1}

做本案例时,发现对于方法反射的获取,要基于结构体指针才可以,目前不解,需要在深入下。

我们获取的属性和方法分别属于 reflect.StructFieldreflect.Method 类型,若需要接续获取属性字段或方法的信息,可以使用该类型定义的方法完成。定义如下,供参考:

type StructField struct {
Name string // 字段名
PkgPath string // 非导出字段的包路径,对导出字段该字段为""
Type Type // 字段类型
Tag StructTag // 字段标签
Offset uintptr // 字段在结构体中的字节偏移量
Index []int // 用于Type.FieldByIndex时的索引切片
Anonymous bool // 是否匿名字段
} type Method struct {
Name string // 方法名
PkgPath string // 非导出方法的包路径,对导出方法该字段为""
Type Type // 方法类型
Func Value // 方法值
Index int // 方法索引
}

也支持:

  • FieldByName(name string) (StructField, bool),通过字段名字确定字段的反射
  • MethodByName(string) (Method, bool),通过方法名字确定方法的反射。

6 结构体标签

结构体标签,Struct Tag,指的是为字段增加额外的属性,利用反射可获取到这些属性,进而完成特定操作。例如:

type Stu struct {
Name string `json:"name" bson:"name"`
Sn string `json:"sn" bson:"sn"`
}

字段后反引号包裹的就是字段的标签。上面的标签是一个常用的格式,在做结构体序列化时经常使用。

利用反射获取标签内容,先获取字段,再获取字段上的标签:

type Stu struct {
Name string `json:"j_name" bson:"b_name"`
Sn string `json:"j_sn" bson:"b_sn"`
} func main() {
var v Stu
typeV := reflect.TypeOf(v)
for i, c := 0, typeV.NumField(); i < c; i++ {
fmt.Println(typeV.Field(i).Tag.Get("json"), typeV.Field(i).Tag.Get("bson"))
}
}
// 输出
j_name b_name
j_sn b_sn

标签语法是key:value结构。(也可以字符串,key:value 更长用,信息量更大)。StructField.Tag 可以获取字段的标签,.Get() 方法可以获取具体内容。

演示,利用标签 json 编码我们的结构体对象,需要 encoding/json 包:

type Stu struct {
Name string `json:"j_name" bson:"b_name"`
Sn string `json:"j_sn" bson:"b_sn"`
}
func main() {
var v = Stu{
"Hank",
"Kang-007",
}
json, err := json.Marshal(v)
fmt.Println(string(json), err)
// {"j_name":"Hank","j_sn":"Kang-007"} <nil>
}

注意上面的 json 中的字段,并不是我们的字段Name和Sn,而是标签中定义的j_name, j_sn。json.Marshal 方法就读取了字段的tag,确定了字段的名称。除了字段名称提示外,json.Marshal 还支持 json:"j_sn,string,omitempty" 表示设置名称,类型,忽略空值等操作。

也可利用 json 转换得到结构体对象,继续使用上面的结构体Stu:

var u Stu
if nil == json.Unmarshal(str, &u) {
fmt.Println(u)
// {Hank Kang-007}
}

完!
原文出自:小韩说课
微信关注:小韩说课

Go语言反射之类型反射的更多相关文章

  1. c#反射机制学习和利用反射获取类型信息

    反射(Reflection)是.NET中的重要机制,通过放射,可以在运行时获得.NET中每一个类型(包括类.结构.委托.接口和枚举等)的成员,包括方法.属性.事件,以及构造函数等.还可以获得每个成员的 ...

  2. Go语言学习之6 反射详解

    1.反射: 定义: 反射就是程序能够在运行时检查变量和值,求出它们的类型.                   可以在运行时动态获取变量的相关信息                   Import ( ...

  3. Go语言的接口与反射

    美女图片没啥用,就是为了好看 本文还在完善中... go总体而言是一门比较好入门的语言,许多特性都很精简易懂,但是接口与反射除外.他们真的让人头疼,不知道是自身资质问题还是怎么着,总是觉得很多书上写的 ...

  4. go反射----1类型

    声明:文章内容取自雨痕老师<Go语言学习笔记> 反射( reflect )让我们能在运行期探知对象的类型信息和内存结构,这从一定程度上弥补了静态语言在动态行为上的不足.同时,反射还是元编程 ...

  5. C#反射实现 C# 反射 判断类的延伸类型 使用代码生成工具Database2Sharp快速生成工作流模块控制器和视图代码 C# ADO.NET的SqlDataReader对象,判断是否包含指定字段 页面中添加锚点的几种方式 .net 简单实用Log4net(多个日志配置文件) C# 常用小点

    C#反射实现   一.反射概念: 1.概念: 反射,通俗的讲就是我们在只知道一个对象的内部而不了解内部结构的情况下,通过反射这个技术可以使我们明确这个对象的内部实现. 在.NET中,反射是重要的机制, ...

  6. GO_09:GO语言基础之reflect反射

    反射reflection 1. 反射可以大大的提高程序的灵活性,使得 interface{} 有更大的发挥余地 2. 反射使用 TypeOf 和 ValueOf 函数从接口中获取目标对象信息 3. 反 ...

  7. GO语言基础之reflect反射

    反射reflection 1. 反射可以大大的提高程序的灵活性,使得 interface{} 有更大的发挥余地 2. 反射使用 TypeOf 和 ValueOf 函数从接口中获取目标对象信息 3. 反 ...

  8. Java---类反射(1)---类反射入门和基础

    什么是类反射 ☆什么是反射 JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法和属性:这种动态获取的信息以及动态调用对象的方 ...

  9. Java 反射机制[Method反射]

    Java 反射机制[Method反射] 接着上一篇Java 反射机制[Field反射],通过调用Person类的setName方法将obj的name字段的Value设置为"callPerso ...

随机推荐

  1. July 31st 2017 Week 31st Monday

    Elegance is the only beauty that never fades. 优雅是唯一不会褪色的美. Even the most beautiful apperace would be ...

  2. December 08th 2016 Week 50th Thursday

    Life is a test and this world a place of trial. 人生是一场考试,这个世界就是考场. I have not passed the test yet. I ...

  3. 离散对数&&大步小步算法及扩展

    bsgs algorithm ax≡b(mod n) 大步小步算法,这个算法有一定的局限性,只有当gcd(a,m)=1时才可以用 原理 此处讨论n为素数的时候. ax≡b(mod n)(n为素数) 由 ...

  4. 揭秘:C++编译器的函数编译流程

    http://www.cnblogs.com/zhenjing/archive/2010/10/20/1856309.html C++中的类型查找过程相对简单,基本上就是名字查找,这里不再介绍. 对于 ...

  5. wireshark:no interface can be used for capturing in this system with the current configuration

    在虚拟机unbuntu中,进行wireshark抓包,出现:no interface can be used for capturing in this system with the current ...

  6. git回滚线上代码

        由于之前自己推代码的时候操作失误,push代码的时候没有push到线上的dev分支,而是push到了线上master分支(主要是因为没有在命令后写分支名,直接推到默认master分支上了),覆 ...

  7. BZOJ1334:[Baltic2008]Elect(背包DP)

    Description N个政党要组成一个联合内阁,每个党都有自己的席位数. 现在希望你找出一种方案,你选中的党的席位数要大于总数的一半,并且联合内阁的席位数越多越好. 对于一个联合内阁,如果某个政党 ...

  8. GPS-Graph Processing System Graph Coloring算法分析 (三)

        HamaWhite 原创,转载请注明出处!欢迎大家增加Giraph 技术交流群: 228591158     Graph coloring is the problem of assignin ...

  9. Kali-linux使用假冒令牌

    使用假冒令牌可以假冒一个网络中的另一个用户进行各种操作,如提升用户权限.创建用户和组等.令牌包括登录会话的安全信息,如用户身份识别.用户组和用户权限.当一个用户登录Windows系统时,它被给定一个访 ...

  10. [Python 多线程] RLock可重入锁 (九)

    RLock 可重复锁,是线程相关的锁.同样是线程相关的还有threading.local. 线程A获得可重用锁,并可以多次成功获取,不会阻塞.最后要再线程A中和acquire次数相同的release. ...