GO语言基础之reflect反射
反射reflection
1. 反射可以大大的提高程序的灵活性,使得 interface{} 有更大的发挥余地
2. 反射使用 TypeOf 和 ValueOf 函数从接口中获取目标对象信息
3. 反射会将匿名字段作为独立字段(匿名字段本质)
4. 想要利用反射修改对象状态,前提是 interface.data 是 settable,即 pointer-interface
5. 通过反射可以“动态”调用方法
示例一:
举例说明反射使用 TypeOf 和 ValueOf 来取得传入类型的属性字段于方法

package main import (
"fmt"
"reflect"
) //定义一个用户结构体
type User struct {
Id int
Name string
Age int
} //为接口绑定方法
func (u User) Hello() {
fmt.Println("Hello World.")
} //定义一个可接受任何类型的函数(空接口的使用规则)
func Info(o interface{}) {
t := reflect.TypeOf(o) //获取接受到到接口到类型
fmt.Println("Type:", t.Name()) //打印对应类型到名称(这是reflect中自带到) //Kind()方法是得到传入类型到返回类型;下面执行判断传入类型是否为一个结构体
if k := t.Kind(); k != reflect.Struct {
fmt.Println("传入的类型有误,请检查!")
return
} v := reflect.ValueOf(o) //获取接受到到接口类型包含到内容(即其中到属性字段和方法)
fmt.Println("Fields:") //如何将其中到所有字段和内容打印出来呢?
/**
通过接口类型.NumField 获取当前类型所有字段个数
*/
for i := 0; i < t.NumField(); i++ {
f := t.Field(i) //取得对应索引的字段
val := v.Field(i).Interface() //取得当前字段对应的内容
fmt.Printf("%6s: %v = %v\n", f.Name, f.Type, val)
}
/**
通过接口类型.NumMethod 获取当前类型所有方法的个数
*/
fmt.Println("Method:")
for i := 0; i < t.NumMethod(); i++ {
m := t.Method(i) //取得对应索引的方法
fmt.Printf("%6s: %v\n", m.Name, m.Type)
}
} func main() {
u := User{1, "OK", 12}
Info(u)
//Info(&u) 如果传入的是结构体的地址或指针(pointer-interface),那么在Info函数中的Kind方法进行判断时就会被拦截返回
}

运行结果如下:
|
1
2
3
4
5
6
7
|
Type: UserFields: Id: int = 1 Name: string = OK Age: int = 12Method: Hello: func(main.User) |
示例二:
如何通过反射得道结构当中匿名或者嵌入字段

package main import (
"fmt"
"reflect"
) //定义一个用户结构体
type User struct {
Id int
Name string
Age int
} type Manager struct {
User //定义了一个匿名引用
title string
} func main() {
m := Manager{User: User{1, "OK", 15}, title: "123"}
t := reflect.TypeOf(m) //取得类型中的字段是否为匿名字段
fmt.Printf("%6v\n", t.Field(0))
/**
打印内容:{User main.User 0 [ 0] true},其中true表示是匿名类型
那么想要取匿名类型中的字段又该怎么取呢?这里需要使用序号组,传入要取的切片即可
*/
fmt.Printf("%v\n", t.FieldByIndex([]int{0, 0}))
/**
其中上面切片传入的是{0, 0},
1. 第一个0表示当前结构Manager取匿名User是第一个即为0
2. 第二个0表示取得的结构User中要取第一个元素Id相对于User来说也是第一个即为0,如果要取Name则需传入[]int{0, 1}
那么既然可以取出来内容,那么我们就可以尝试着进行修改,怎么做呢?
*/
tchage := reflect.ValueOf(&m) //想要修改和我们之前所说的传入值类型和指针类型是一致的,要想修改需要传入对应指针类型
tchage.Elem().FieldByIndex([]int{0, 0}).SetInt(999) //传入指针需要通过 .Elem() 来取得对应的值内容,之后再想取哪个再继续使用序号组
fmt.Println(tchage.Elem().FieldByName("title"))
fmt.Println(tchage)
}

运行结果:
|
1
2
3
4
|
{ User main.User 0 [ 0] true}{Id int 0 [0] false}123&{{999 OK 15} 123} |
示例三:
那么让我们来写一个比较完整的通过反射修改结构体内部字段内容

package main import (
"fmt"
"reflect"
) //定义一个用户结构体
type User struct {
Id int
Name string
Age int
} func main() {
u := User{1, "OK", 13}
fmt.Println(u)
Set(&u)
fmt.Println(u)
} //定义一个可以接受任何类型的空接口
func Set(o interface{}) {
v := reflect.ValueOf(o)
//通过反射修改类型中的内容需要传入指针,为了防止传入有误故在这里进行相关过滤验证判断(这前这快是已经说过的)
if v.Kind() == reflect.Ptr && !v.Elem().CanSet() {
//reflect.Ptr对应为指针类型;v.Elem().CanSet()取得对应地址下的内容并查看其是否可以进行修改
fmt.Println("传入的类型有误,请检查!")
return
} else {
v = v.Elem() //将实际对象(包含详情内容)进行赋值
} f := v.FieldByName("Name")
f1 := v.FieldByName("Id1")
if !f.IsValid() { //判断通过名称获取得到到内容是否为空值
fmt.Println("没有Name对应属性字段")
return
}
if !f1.IsValid() {
fmt.Println("没有Id1对应属性字段")
}
if f.Kind() == reflect.String {
f.SetString("HelloWorld")
}
}

运行结果:
|
1
2
3
|
{1 OK 13}没有Id1对应属性字段{1 HelloWorld 13} |
示例四:
那么让我们来写一个比较完整的通过反射对方法等动态调用

package main import (
"fmt"
"reflect"
) //定义一个用户结构体
type User struct {
Id int
Name string
Age int
} //为User绑定方法
func (u User) HelloDisplay(name string) {
fmt.Println("Hello", name, " my name is ", u.Name)
} func main() {
u := User{1, "OK", 29}
u.HelloDisplay("jack") //正常调用 /**
以下方式为反射调用,最优到代码写法就是新写一个方法且在开始是通过kind判断类型是否正确且需要判断有没有对应方法等
*/
v := reflect.ValueOf(u) //通过反射得到类型内容
methodV := v.MethodByName("HelloDisplay") //通过方法名称得道方法实体
args := []reflect.Value{reflect.ValueOf("jack")} //设置反射传入的参数
methodV.Call(args)
}

运行结果:
|
1
2
|
Hello jack my name is OKHello jack my name is OK |
GO语言基础之reflect反射的更多相关文章
- GO_09:GO语言基础之reflect反射
反射reflection 1. 反射可以大大的提高程序的灵活性,使得 interface{} 有更大的发挥余地 2. 反射使用 TypeOf 和 ValueOf 函数从接口中获取目标对象信息 3. 反 ...
- Go语言基础之反射
Go语言基础之反射 本文介绍了Go语言反射的意义和基本使用. 变量的内在机制 Go语言中的变量是分为两部分的: 类型信息:预先定义好的元信息. 值信息:程序运行过程中可动态变化的. 反射介绍 反射是指 ...
- GO学习-(17) Go语言基础之反射
Go语言基础之反射 本文介绍了Go语言反射的意义和基本使用. 变量的内在机制 Go语言中的变量是分为两部分的: 类型信息:预先定义好的元信息. 值信息:程序运行过程中可动态变化的. 反射介绍 反射是指 ...
- 黑马程序猿————Java基础日常笔记---反射与正則表達式
------Java培训.Android培训.iOS培训..Net培训.期待与您交流! ------- 黑马程序猿----Java基础日常笔记---反射与正則表達式 1.1反射 反射的理解和作用: 首 ...
- Java基础13:反射与注解详解
Java基础13:反射与注解详解 什么是反射? 反射(Reflection)是Java 程序开发语言的特征之一,它允许运行中的 Java 程序获取自身的信息,并且可以操作类或对象的内部属性. Orac ...
- Go 语言基础
Go 语言基础 特点 常用命令 go run 直接运行, 不会生成可执行文件 go build 生成可执行文件, 推荐 加分特点 UTF-8编码 高并发: go 关键字 管道: pipe := mak ...
- goweb-go语言基础
go语言基础 虽然这本书是讲goweb,但还是吧go语言基础过了一遍,由于我之前已经对go语言基础做了一遍系统的学习,这里就当简单回顾一下,不再写过多笔记了,之前的写的博客都有基础知识,O(∩_∩)O ...
- GO学习-(20) Go语言基础之单元测试
Go语言基础之单元测试 不写测试的开发不是好程序员.我个人非常崇尚TDD(Test Driven Development)的,然而可惜的是国内的程序员都不太关注测试这一部分. 这篇文章主要介绍下在Go ...
- Go语言基础之结构体
Go语言基础之结构体 Go语言中没有“类”的概念,也不支持“类”的继承等面向对象的概念.Go语言中通过结构体的内嵌再配合接口比面向对象具有更高的扩展性和灵活性. 类型别名和自定义类型 自定义类型 在G ...
随机推荐
- Xshell拖拽上传文件插件
lrzsz是一款在linux里可代替ftp上传和下载的程序.在linux中支持直接拖拽上传的插件:同时也支持rz和sz进行命令上传和下载. 插件安装 yum -y install lrzsz 上传(r ...
- join和 Daemon守护线程
一.前言 一个程序至少有一个主线程,主线程启动子线程后,它们之间并没有隶属关系.主线程和子线程执行是并行的,相互独立.主线程执行完毕后默认不等子线程执行结束就接着往下走了,如果有其他程序就会运行另外的 ...
- fiddler 手机装证书
1: 打开浏览器的地址 输入fiddler 右上方的地址 172.21.14.197:8888 2: 然后点击页面里的 FiddlerRoot certificate 3: 最后验证下 iOS 1 ...
- 从DOS时代至移动互联网的技术路线回顾
从DOS时代至移动互联网的技术路线回顾 Normal 0 7.8 磅 0 2 false false false EN-US ZH-CN X-NONE /* Style Definitions */ ...
- hdu 4403 爆搜
题意:给一串数字,在其间加入若干加号和一个等号,问使等式成立的方案总数 if the digits serial is "1212", you can get 2 equation ...
- Java混剪音频
分享一个之前看过的程序,可以用来剪辑特定长度的音频,将它们混剪在一起,思路如下: 1.使用 FileInputStream 输入两个音频 2.使用 FileInputStream的skip(long ...
- BZOJ 2142 礼物 组合数学 CRT 中国剩余定理
2142: 礼物 Time Limit: 10 Sec Memory Limit: 259 MBSubmit: 1450 Solved: 593[Submit][Status][Discuss] ...
- poj3268 Silver Cow Party(农场派对)
题目描述 原题来自:USACO 2007 Feb. Silver N(1≤N≤1000)N (1 \le N \le 1000)N(1≤N≤1000) 头牛要去参加一场在编号为 x(1≤x≤N)x(1 ...
- Linux下Qt安装
1.下载qt-everywhere-opensource-src4.7.2.tar.gz(http://download.qt.io/archive/qt/4.7),并解压在/opt目录下,文件名为q ...
- SlickSafe.NET 开源权限框架开发指南
前言:本文适用于快速搭建权限系统的用户,尤其适用于希望有良好定义的权限模型建立:系统解决方案是在基于角色访问控制(RBAC)策略基础上的权限访问模型实现,主要完成了后台权限验证逻辑和前端权限数据验证的 ...