Golang--反射

1、概念

反射可以做什么?

  • 反射可以在运行时动态获取变量的各种信息,比如变量的类型,类别等信息
  • 如果是结构体变量,还可以获取到结构体本身的信息(包括结构体的字段、方法)
  • 通过反射,可以修改变量的值,可以调用关联的方法。
  • 使用反射,需要import("reflect")



反射相关的函数

  • reflect.TypeOf(变量名),获取变量的类型,返回reflect.Type类型(结构体类型)
  • reflect.ValueOf变量名),获取变量的值,返回reflect.Value类型(reflect.Value是一个结构体类型),通过reflect.Value,可以获取到关于该变量的很多信息。

type Type interface {
    // Kind返回该接口的具体分类
    Kind() Kind
    // Name返回该类型在自身包内的类型名,如果是未命名类型会返回""
    Name() string
    // PkgPath返回类型的包路径,即明确指定包的import路径,如"encoding/base64"
    // 如果类型为内建类型(string, error)或未命名类型(*T, struct{}, []int),会返回""
    PkgPath() string
    // 返回类型的字符串表示。该字符串可能会使用短包名(如用base64代替"encoding/base64")
    // 也不保证每个类型的字符串表示不同。如果要比较两个类型是否相等,请直接用Type类型比较。
    String() string
    // 返回要保存一个该类型的值需要多少字节;类似unsafe.Sizeof
    Size() uintptr
    // 返回当从内存中申请一个该类型值时,会对齐的字节数
    Align() int
    // 返回当该类型作为结构体的字段时,会对齐的字节数
    FieldAlign() int
    // 如果该类型实现了u代表的接口,会返回真
    Implements(u Type) bool
    // 如果该类型的值可以直接赋值给u代表的类型,返回真
    AssignableTo(u Type) bool
    // 如该类型的值可以转换为u代表的类型,返回真
    ConvertibleTo(u Type) bool
    // 返回该类型的字位数。如果该类型的Kind不是Int、Uint、Float或Complex,会panic
    Bits() int
    // 返回array类型的长度,如非数组类型将panic
    Len() int
    // 返回该类型的元素类型,如果该类型的Kind不是Array、Chan、Map、Ptr或Slice,会panic
    Elem() Type
    // 返回map类型的键的类型。如非映射类型将panic
    Key() Type
    // 返回一个channel类型的方向,如非通道类型将会panic
    ChanDir() ChanDir
    // 返回struct类型的字段数(匿名字段算作一个字段),如非结构体类型将panic
    NumField() int
    // 返回struct类型的第i个字段的类型,如非结构体或者i不在[0, NumField())内将会panic
    Field(i int) StructField
    // 返回索引序列指定的嵌套字段的类型,
    // 等价于用索引中每个值链式调用本方法,如非结构体将会panic
    FieldByIndex(index []int) StructField
    // 返回该类型名为name的字段(会查找匿名字段及其子字段),
    // 布尔值说明是否找到,如非结构体将panic
    FieldByName(name string) (StructField, bool)
    // 返回该类型第一个字段名满足函数match的字段,布尔值说明是否找到,如非结构体将会panic
    FieldByNameFunc(match func(string) bool) (StructField, bool)
    // 如果函数类型的最后一个输入参数是"..."形式的参数,IsVariadic返回真
    // 如果这样,t.In(t.NumIn() - 1)返回参数的隐式的实际类型(声明类型的切片)
    // 如非函数类型将panic
    IsVariadic() bool
    // 返回func类型的参数个数,如果不是函数,将会panic
    NumIn() int
    // 返回func类型的第i个参数的类型,如非函数或者i不在[0, NumIn())内将会panic
    In(i int) Type
    // 返回func类型的返回值个数,如果不是函数,将会panic
    NumOut() int
    // 返回func类型的第i个返回值的类型,如非函数或者i不在[0, NumOut())内将会panic
    Out(i int) Type
    // 返回该类型的方法集中方法的数目
    // 匿名字段的方法会被计算;主体类型的方法会屏蔽匿名字段的同名方法;
    // 匿名字段导致的歧义方法会滤除
    NumMethod() int
    // 返回该类型方法集中的第i个方法,i不在[0, NumMethod())范围内时,将导致panic
    // 对非接口类型T或*T,返回值的Type字段和Func字段描述方法的未绑定函数状态
    // 对接口类型,返回值的Type字段描述方法的签名,Func字段为nil
    Method(int) Method
    // 根据方法名返回该类型方法集中的方法,使用一个布尔值说明是否发现该方法
    // 对非接口类型T或*T,返回值的Type字段和Func字段描述方法的未绑定函数状态
    // 对接口类型,返回值的Type字段描述方法的签名,Func字段为nil
    MethodByName(string) (Method, bool)
    // 内含隐藏或非导出方法
}

 

2、对基本数据类型反射

反射相关的函数

  • reflect.TypeOf(变量名),获取变量的类型,返回reflect.Type类型(结构体类型)
  • reflect.ValueOf变量名),获取变量的值,返回reflect.Value类型(reflect.Value是一个结构体类型),通过reflect.Value,可以获取到关于该变量的很多信息。
package main

import(
	"fmt"
	"reflect"
)

func testReflect(i interface{}){
	//1、调用TypeOf()函数,返回reflect.Type类型的变量
	reType := reflect.TypeOf(i)
	fmt.Println("reType=",reType)  //int,但不代表reType是int,只是代表reType是int的变量,reType类型是reflect.Type
	fmt.Printf("reType的类型是:%T\n",reType) //*reflect.Type

	//2、调用VauleOf()函数,返回reflect.Value类型的变量
	reValue := reflect.ValueOf(i)
	fmt.Println("reValue=",reValue) // 119,但不代表reValue是119,只是代表reValue是119的变量,reValue类型是reflect.Value
	fmt.Printf("reValue的类型是:%T\n",reValue) //reflect.Value

	//如果想获取reValue的数值,要调用Int()方法:返回reValue持有的的数值
	num2 := 80 + reValue.Int() //Int()方法返回reValue持有的的数值--直接用reValue是不行的,因为reValue是reflect.Value类型的变量,类型不匹配不能直接用,需要调用Int()方法
	fmt.Println("num2=",num2) // 199

	//reValue转成空接口
	i2 := reValue.Interface()
	//类型断言
	n := i2.(int)
	n2 := n + 100
	fmt.Println(n2)
}

func main(){
	//对基本数据类型进行反射
	//定义一个int类型的变量
	var num int = 119
	//获取变量的类型	
	testReflect(num)
}

3、 对结构体进行反射

与基本数据类型同理

package main

import(
	"fmt"
	"reflect"
)

type Student struct{
	Name string
	Age int
}


func testReflect(i interface{}){
	//1、调用TypeOf()函数,返回reflect.Type类型的变量
	reType := reflect.TypeOf(i)
	fmt.Println("reType=",reType) //main.Student
	fmt.Printf("reType的类型是:%T\n",reType) //*reflect.rtype

	//2、调用VauleOf()函数,返回reflect.Value类型的变量
	reValue := reflect.ValueOf(i)
	fmt.Println("reValue=",reValue) //{张三 18}
	fmt.Printf("reValue的类型是:%T\n",reValue) //reflect.Value 

	//reValue转成空接口
	i2 := reValue.Interface()
	//类型断言
	n,flag := i2.(Student)
	if flag{//断言成功
		fmt.Println(n.Name,' ',n.Age) // 张三 18
	}else{//断言失败
		fmt.Println("类型断言失败")
	}
}

func main(){
	//对结构体数据类型进行反射
	//定义一个结构体类型的变量
	stu := Student{
		Name: "张三",
		Age: 18,
	}

	//获取变量的类型	
	testReflect(stu)
}

4、获取变量的类别

类型和类别的区别:

  • 类别是一个大的方向:例如不同的结构体类型都属于struct这个结构体类别
  • 类型是具体某一个类型:例如不同的结构体类型属于不同的类型
     


 

package main

import(
	"fmt"
	"reflect"
)

type Student struct{
	Name string
	Age int
}


func testReflect(i interface{}){
	//1、调用TypeOf()函数,返回reflect.Type类型的变量
	reType := reflect.TypeOf(i)

	//2、调用VauleOf()函数,返回reflect.Value类型的变量
	reValue := reflect.ValueOf(i)

	//3、获取变量的类别--Type
	fmt.Println("类别:",reType.Kind()) //类别:struct

	//4、获取变量的类别--Value
	fmt.Println("类别:",reValue.Kind()) //类别:struct

	//获取变量的类型:
	//reValue转成空接口
	i2 := reValue.Interface()
	//类型断言:
	n,flag := i2.(Student)
	if flag{
		fmt.Printf("类型:%T",n) // 类型:main.Student
	}
}

func main(){
	//对结构体数据类型进行反射
	//定义一个结构体类型的变量
	stu := Student{
		Name: "张三",
		Age: 18,
	}

	//获取变量的类型	
	testReflect(stu)
}

 5、通过反射修改变量

package main

import(
	"fmt"
	"reflect"
)

type Student struct{
	Name string
	Age int
}


func testReflect(i interface{}){
	reValue := reflect.ValueOf(i)

	//通过SetInt()修改值
	reValue.Elem().SetInt(200)
}

func main(){
	var num int = 100
	testReflect(&num) //传入的是地址,才能修改值
	
	fmt.Println(num) // 200
}

6、通过反射操作结构体的属性和方法

package main
import (
	"fmt"
	"reflect"
)

type Student struct{
	Name string
	Age int
}
func (s Student) APrint(){
	fmt.Println("Name:",s.Name)
	fmt.Println("Age:",s.Age)
}

func (s Student) BGetSum(n1,n2 int) int{
	return n1 + n2
}

func (s *Student) CSet(name string,age int){
	s.Name = name
	s.Age = age
}

//定义函数操作结构体进行反射操作
func TestStudentReflect(a interface{}){
	//a转成reflect.Value
	val := reflect.ValueOf(a)

	//通过reflect.Value类型操作结构体内部的字段:
	n1 := val.NumField() // 获取结构体字段的数量
	fmt.Println(n1)
	//通过遍历--获取具体的字段
	for i := 0; i < n1; i++{
		fmt.Printf("第%d个字段的值是:%v\n",i,val.Field(i)) //获取第i个字段的值
	}

	//通过reflect.Value类型操作结构体内部的方法:
	n2 := val.NumMethod() // 获取结构体方法的数量
	fmt.Println(n2)

	//调用方法
	//调用方法,方法的首字母必须大写才能有对应的反射的访问权限
	//方法的顺序按照ASCII码表的顺序进行排序的,对应索引从0开始,依次类推
	val.Method(0).Call(nil) //调用第0个方法

	var params []reflect.Value
	params = append(params,reflect.ValueOf(11))
	params = append(params,reflect.ValueOf(19))
	result := val.Method(1).Call(params)
	fmt.Println(result[0].Int()) //30
}

//通过反射改变结构体的值
func TestStudentReflect2(a interface{}){
	//a转成reflect.Value
	val := reflect.ValueOf(a)
	
	n := val.Elem().NumField() // 获取结构体字段的数量
	fmt.Println(n)

	//直接修改字段的值
	val.Elem().Field(0).SetString("李四")
	val.Elem().Field(1).SetInt(21)

	//调用上面定义的方法--Cset()
	var params []reflect.Value
	params = append(params,reflect.ValueOf("王五"))
	params = append(params,reflect.ValueOf(22))

	val.Method(2).Call(params) //CSet方法
}

func main(){
	a := Student{
		Name: "张三",
		Age: 20,
	}
	TestStudentReflect(a)
	fmt.Println("")

	TestStudentReflect2(&a)
	fmt.Println(a) //{王五 22}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/912067.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

释放专利力量:Patently 如何利用向量搜索和 NLP 简化协作

作者&#xff1a;来自 Elastic Matt Scourfield, Andrew Crothers, Brian Lambert 组织依靠知识产权 (IP) 来推动创新、保持竞争优势并创造收入来源。对于希望将新产品推向市场的公司来说&#xff0c;弄清楚谁拥有哪些专利是一项必不可少的能力。搜索数百万项专利可能既困难又耗…

[Linux] 进程等待 | 进程替换

&#x1fa90;&#x1fa90;&#x1fa90;欢迎来到程序员餐厅&#x1f4ab;&#x1f4ab;&#x1f4ab; 主厨&#xff1a;邪王真眼 主厨的主页&#xff1a;Chef‘s blog 所属专栏&#xff1a;青果大战linux 总有光环在陨落&#xff0c;总有新星在闪烁 我有一个朋友&#x…

【EasyExcel】EasyExcel导出表格包含合计行、自定义样式、自适应列宽

目录 0 EasyExcel简介1 Excel导出工具类设置自定义表头样式设置自适应列宽添加合计行 2 调用导出工具类导出Excel表3 测试结果 0 EasyExcel简介 在数据处理和报表生成的过程中&#xff0c;Excel是一个非常常用的工具。特别是在Java开发中&#xff0c;EasyExcel库因其简单高效而…

深度优先搜索之全排列问题(C语言版)

本文的一些参考&#xff1a; DFS (深度优先搜索) 算法详解 模板 例题&#xff0c;这一篇就够了_dfs算法-CSDN博客 首先把深度优先搜索算法的基本概论摆出来 深度优先搜索算法&#xff08;Depth First Search&#xff0c;简称DFS&#xff09;&#xff1a; 一种用于遍历或搜…

【Docker】自定义网络:实现容器之间通过域名相互通讯

文章目录 一. 默认网络&#xff1a;docker0网络的问题二. 自定义网络三. nginx容器指之间通过主机名进行内部通讯四. redis集群容器&#xff08;跳过宿主机&#xff09;内部网络通讯1. 集群描述2. 基于bitnami镜像的环境变量快速构建redis集群 一. 默认网络&#xff1a;docker0…

Serverless+AI,前沿技术

大家好&#xff0c;我是袁庭新。如果想在未来成为一名合格且具备前瞻视野的软件开发工程师&#xff0c;新兴且热门的技术领域都是需要去了解的&#xff08;例如包括ServerlessAI、AI可观测性、以及AI原生应用架构&#xff09;&#xff0c;并且在参加工作前尽可能去系统学习掌握…

开放式耳机如何选择?五款千万不能错过的开放式耳机机型推荐

在这里我先做一个行业的知识科普&#xff0c;目前市场上有超过80%的品牌&#xff0c;都是非专业的开放式耳机品牌&#xff0c;也就是跨界大牌或者网红品牌&#xff0c;这些品牌由于没有开放式声学的技术沉淀&#xff0c;在制作开放式耳机的时候&#xff0c;通常都是直接套用传统…

力扣17-电话号码的数字组合

力扣17-电话号码的数字组合 思路代码 题目链接 思路 原题&#xff1a; 给定一个仅包含数字 2-9 的字符串&#xff0c;返回所有它能表示的字母组合。答案可以按 任意顺序 返回。 给出数字到字母的映射如下&#xff08;与电话按键相同&#xff09;。注意 1 不对应任何字母。 输…

鸿蒙进阶篇-剩余和展开、简单和复杂类型

“在科技的浪潮中&#xff0c;鸿蒙操作系统宛如一颗璀璨的新星&#xff0c;引领着创新的方向。作为鸿蒙开天组&#xff0c;今天我们将一同踏上鸿蒙基础的探索之旅&#xff0c;为您揭开这一神奇系统的神秘面纱。” 各位小伙伴们我们又见面了,我就是鸿蒙开天组,下面让我们进入今…

【复平面】-复数相乘的几何性质

文章目录 从数学上证明1. 计算乘积 z 1 ⋅ z 2 z_1 \cdot z_2 z1​⋅z2​2. 应用三角恒等式3. 得出结果 从几何角度证明1.给出待乘的复数 u i u_i ui​2.给出任意复数 l l l3.复数 l l l 在不同坐标轴下的表示图 首先说结论&#xff1a; 在复平面中&#xff0c;两个复数&a…

【EMNLP2024】基于多轮课程学习的大语言模型蒸馏算法 TAPIR

近日&#xff0c;阿里云人工智能平台PAI与复旦大学王鹏教授团队合作&#xff0c;在自然语言处理顶级会议EMNLP 2024 上发表论文《Distilling Instruction-following Abilities of Large Language Models with Task-aware Curriculum Planning》。文章提出了一个名为 TAPIR 的知…

Web服务nginx基本实验

安装软件&#xff1a; 启动服务&#xff1a; 查看Nginx服务器的网络连接信息&#xff0c;监听的端口&#xff1a; 查看默认目录&#xff1a; 用Windows访问服务端192.168.234.111的nginx服务&#xff1a;&#xff08;防火墙没有放行nginx服务&#xff0c;访问不了&#xff09; …

github使用基础

要通过终端绑定GitHub账号并进行文件传输&#xff0c;你需要使用Git和SSH密钥来实现安全连接和操作。以下是一个基本流程&#xff1a; 设置GitHub和SSH 检查Git安装 通过终端输入以下命令查看是否安装Git&#xff1a; bash 复制代码 git --version配置Git用户名和邮箱 bash …

excel常用技能

1.基础技能 1.1 下拉框设置 a. 选中需要设置的列或单元格&#xff0c;数据 ---》 数据验证 b.验证条件 ---> 序列&#xff08;多个值逗号隔开&#xff09; 2.函数 2.1 统计函数-count a.count(区域&#xff0c;区域&#xff0c;......) 统计数量&#xff0c;只针…

Flipper Zero BadUSB反弹shell

Flipper Zero BadUSB反弹shell 前置知识点&#xff1a; Flipper Zero BadUSB 以及其他几个 BadUSB 设备使用用 DuckyScript 编写的有效负载。一种简单的脚本语言&#xff0c;用于执行导致键盘注入攻击的击键。 步骤 创建rev_shell_win.txt文件,并将其拖到badusb文件夹中. 相…

【GPTs】Email Responder Pro:高效生成专业回复邮件

博客主页&#xff1a; [小ᶻZ࿆] 本文专栏: AIGC | GPTs应用实例 文章目录 &#x1f4af;GPTs指令&#x1f4af;前言&#x1f4af;Email Responder Pro主要功能适用场景优点缺点 &#x1f4af;小结 &#x1f4af;GPTs指令 Email Craft is a specialized assistant for cra…

检测敏感词功能

今天策划给我一个任务 —— 检测昵称中是否含有敏感词功能&#xff0c;然后丢给我两个压缩包&#xff0c;我解压一看&#xff1a; 有的txt文件是一行一个词&#xff1a; 有的txt文件是按逗号分隔开&#xff1a; 不管是什么格式的总之量非常多&#xff0c;把我这辈子脏话都囊括…

【SpringBoot】19 文件/图片下载(MySQL + Thymeleaf)

Git仓库 https://gitee.com/Lin_DH/system 介绍 从 MySQL 中&#xff0c;下载保存的 blob 格式的文件。 代码实现 第一步&#xff1a;配置文件 application.yml spring:jackson:date-format: yyyy-MM-dd HH:mm:sstime-zone: GMT8datasource:driver-class-name: com.mysql.…

Coppelia Sim (v-REP)仿真 机器人3D相机手眼标定与实时视觉追踪 (三)

使用标定好的结果进行跟踪标定板的位置 坐标转换的步骤为&#xff1a; 1.图像坐标点转到相机坐标系下的点 2.相机坐标系下的点转为夹爪坐标系下的点 3.夹爪坐标系下的点转为机械手极坐标系下的点 跟踪的方式 1.采用标定板的第一个坐标点作为跟踪点 3.机器人每次移动到该点位&a…

easyui +vue v-slot 注意事项

https://www.jeasyui.com/demo-vue/main/index.php?pluginDataGrid&themematerial-teal&dirltr&pitemCheckBox%20Selection&sortasc 接口说明 <template><div><h2>Checkbox Selection</h2><DataGrid :data"data" style&…