Go语言之反射(反射的简单使用,原理)

news2024/11/16 22:20:52

一、反射的基础

1.什么是反射

  • Go语言中,反射的机制就是在运行的时候,可以获取到其变量的类型和值,且可以对其类型和值进行检查,对其值进行修改。
  • 即在不知道具体的类型的情况下,可以用反射机制来查看变量类型、更新变量的值。
  • Go中反射主要涉及到两个概念:Type和Value。对所有的接口进行反射时,都可以得到一个包含Type和Value的信息结构,Type是反射的这个变量本身的类型信息,Value是反射的这个变量本身的值信息。

2.反射的使用

(1) 获取到变量类型

  • 方法:reflect.TypeOf(x)
  • 作用:获取变量x的类型信息
    package main
    
    import (
    	"fmt"
    	"reflect"
    )
    
    func main(){
    	x := 3.1415
    	y := 3
    	z := "sheena"
    	u := true
    
    	fmt.Println("x的type为:", reflect.TypeOf(x))
    	fmt.Println("y的type为:", reflect.TypeOf(y))
    	fmt.Println("z的type为:", reflect.TypeOf(z))
    	fmt.Println("u的type为:", reflect.TypeOf(u))
    }
    
    结果:在这里插入图片描述

(2) 进行变量类型判断

  • 方法:xtype := reflect.TypeOf(x) xkind := xtype.Kind() if xkind == reflect.Type
  • 作用:先调用TypeOf方法获取到x的类型信息,再获取到类型信息中的类型,与类型进行判断
    package main
    
    import (
    	"fmt"
    	"reflect"
    )
    
    func main(){
    	x := 3.1415
    	y := 3
    	z := "sheena"
    	u := true
    
    	//获取变量的类型
    	xtype := reflect.TypeOf(x)
    	ytype := reflect.TypeOf(y)
    	ztype := reflect.TypeOf(z)
    	utype := reflect.TypeOf(u)
    
    	//Kind()方法的返回结果主要是用来进行类型判断的
    	xkind := xtype.Kind()
    	ykind := ytype.Kind()
    	zkind := ztype.Kind()
    	ukind := utype.Kind()
    
    	//对x进行类型判断
    	if xkind == reflect.String{
    		fmt.Println("x的type是string")
    	}else if xkind == reflect.Float64{
    		fmt.Println("x的type是float64")
    	}
    
    	//对y进行类型判断
    	if ykind == reflect.Float64{
    		fmt.Println("y的type是Float64")
    	}else if ykind == reflect.Int{
    		fmt.Println("y的type是int")
    	}
    
    	//对z进行类型判断
    	if zkind == reflect.Float64{
    		fmt.Println("z的type是Float64")
    	}else if zkind == reflect.String{
    		fmt.Println("z的type是string")
    	}
    
    	//对u进行类型判断
    	if ukind == reflect.Bool{
    		fmt.Println("u的type是bool")
    	}else if ukind == reflect.String{
    		fmt.Println("u的type是string")
    	}
    }
    
    结果:
    在这里插入图片描述

(3) 获取到变量值

  • 方法:reflect.ValueOf(x)
  • 作用:获取x变量的值
    package main
    
    import (
    	"fmt"
    	"reflect"
    )
    
    func main(){
    	x := 3.1415
    	y := 3
    	z := "sheena"
    	u := true
    
    	fmt.Println("x的value为:", reflect.ValueOf(x))
    	fmt.Println("y的value为:", reflect.ValueOf(y))
    	fmt.Println("z的value为:", reflect.ValueOf(z))
    	fmt.Println("u的value为:", reflect.ValueOf(u))
    }
    
    结果:
    在这里插入图片描述

(4) 修改变量值

  • 方法一:使用具体类型的值的设置方法,如SetString()、SetFloat64()等(可直接接收具体要修改的值)
    package main
    
    import (
    	"fmt"
    	"reflect"
    )
    
    func main(){
    	x := 3.1415
    	y := 3
    	z := "sheena"
    	u := true
    
    	fmt.Println("x修改前的value为:", reflect.ValueOf(x))
    	fmt.Println("y修改前的value为:", reflect.ValueOf(y))
    	fmt.Println("z修改前的value为:", reflect.ValueOf(z))
    	fmt.Println("u修改前的value为:", reflect.ValueOf(u))
    
    	//通过反射传入变量x的地址,并且通过Ele
    	rex := reflect.ValueOf(&x).Elem()
    	rey := reflect.ValueOf(&y).Elem()
    	rez := reflect.ValueOf(&z).Elem()
    	reu := reflect.ValueOf(&u).Elem()
    
    	//判断是否可以修改变量x的值,若可以,则用SetFLoat64()方法进行修改
    	if rex.CanSet(){
    		rex.SetFloat(61.23466)
    		fmt.Println("x修改后的value为:", reflect.ValueOf(x))
    	}else {
    		fmt.Println("该变量不能修改")
    	}
    
    	if rey.CanSet(){
    		rey.SetInt(10000)
    		fmt.Println("y修改后的value为:", reflect.ValueOf(y))
    	}else {
    		fmt.Println("该变量不能修改")
    	}
    
    	if rez.CanSet(){
    		rez.SetString("hello world")
    		fmt.Println("z修改后的value为:", reflect.ValueOf(z))
    	}else{
    		fmt.Println("该变量不能修改")
    	}
    
    	if reu.CanSet(){
    		reu.SetBool(false)
    		fmt.Println("u修改后的value为:", reflect.ValueOf(u))
    	}else {
    		fmt.Println("该变量不能修改")
    	}
    }
    
    结果:
    在这里插入图片描述
  • 方法二:是用Set方法直接设置值(注意set方法接收的是ValueOf的返回值)
    package main
    
    import (
    	"fmt"
    	"reflect"
    )
    
    func main(){
    	x := 3.1415
    	y := 3
    	z := "sheena"
    	u := true
    
    	fmt.Println("x修改前的value为:", reflect.ValueOf(x))
    	fmt.Println("y修改前的value为:", reflect.ValueOf(y))
    	fmt.Println("z修改前的value为:", reflect.ValueOf(z))
    	fmt.Println("u修改前的value为:", reflect.ValueOf(u))
    
    	//通过反射传入变量x的地址,并且通过Ele
    	rex := reflect.ValueOf(&x).Elem()
    	rey := reflect.ValueOf(&y).Elem()
    	rez := reflect.ValueOf(&z).Elem()
    	reu := reflect.ValueOf(&u).Elem()
    
    	//判断是否可以修改变量x的值,若可以,则用Set()方法进行修改
    	if rex.CanSet(){
    		ax := reflect.ValueOf(61.23466) // 使用Set方法修改值,Set方法接收的是ValueOf的返回值
    		rex.Set(ax)
    		fmt.Println("x修改后的value为:", reflect.ValueOf(x))
    	}else {
    		fmt.Println("该变量不能修改")
    	}
    
    	if rey.CanSet(){
    		ay := reflect.ValueOf(10000)// 使用Set方法修改值,Set方法接收的是ValueOf的返回值
    		rey.Set(ay)
    		fmt.Println("y修改后的value为:", reflect.ValueOf(y))
    	}else {
    		fmt.Println("该变量不能修改")
    	}
    
    	if rez.CanSet(){
    		az := reflect.ValueOf("hello world")// 使用Set方法修改值,Set方法接收的是ValueOf的返回值
    		rez.Set(az)
    		fmt.Println("z修改后的value为:", reflect.ValueOf(z))
    	}else{
    		fmt.Println("该变量不能修改")
    	}
    
    	if reu.CanSet(){
    		au := reflect.ValueOf(false)// 使用Set方法修改值,Set方法接收的是ValueOf的返回值
    		reu.Set(au)
    		fmt.Println("u修改后的value为:", reflect.ValueOf(u))
    	}else {
    		fmt.Println("该变量不能修改")
    	}
    }
    
    结果:
    在这里插入图片描述

(5) 获取变量的指针所指向的对象

  • 方法:Elem()方法
  • 作用:获取变量x的指针所指向的对象
    package main
    
    import (
    	"fmt"
    	"reflect"
    )
    
    func main(){
    	x := 3.1415
    	y := 3
    	z := "sheena"
    	u := true
    
    	//传入变量地址
    	px := reflect.ValueOf(&x)
    	py := reflect.ValueOf(&y)
    	pz := reflect.ValueOf(&z)
    	pu := reflect.ValueOf(&u)
    
    	fmt.Println("x的地址是", px)
    	fmt.Println("y的地址是", py)
    	fmt.Println("z的地址是", pz)
    	fmt.Println("u的地址是", pu)
    
    	//通过变量地址获取到变量的值
    	xe := px.Elem()
    	ye := py.Elem()
    	ze := pz.Elem()
    	ue := pu.Elem()
    	
    	fmt.Println("x的值是", xe)
    	fmt.Println("y的值是", ye)
    	fmt.Println("z的值是", ze)
    	fmt.Println("u的值是", ue)
    }
    
    结果:
    在这里插入图片描述

(6) 获取结构体变量的类型和值

package main

import (
	"fmt"
	"reflect"
)

type Stu struct{
	Name string
	Age int
	Sex string
	IsCan bool
}

func main(){
	s1 := Stu{Name: "王一", Age: 18, Sex: "男", IsCan: false}
	s2 := Stu{Name: "王二", Age: 19, Sex: "女", IsCan: true}
	s3 := Stu{Name: "张三", Age: 20, Sex: "男", IsCan: false}

	//反射获取结构体的类型和值
	fmt.Println("s1的类型", reflect.TypeOf(s1))
	fmt.Println("s1的值", reflect.ValueOf(s1))

	fmt.Println("s2的类型", reflect.TypeOf(s2))
	fmt.Println("s2的值", reflect.ValueOf(s2))

	fmt.Println("s3的类型", reflect.TypeOf(s3))
	fmt.Println("s3的值", reflect.ValueOf(s3))


	fmt.Println("TypeOf()和Kind()方法输出的区别")
	fmt.Println("TypeOf(s1):", reflect.TypeOf(s1))
	s1tp := reflect.TypeOf(s1)
	fmt.Println("Kind(s1):", s1tp.Kind())
}

结果:
在这里插入图片描述

二、反射的原理

1.反射如何获取类型信息

  • reflect包中提供TypeOf接口用于获取一个变量的类型信息。它接收一个空接口类型的参数,并返回一个reflect.Type类型的返回值
  • TypeOf接口:
    func TypeOf(i interface{}) Type{
    	eface := *(*emptyInterface)(unsafe.Pointer(&i))
    	return toType(eface.type)
    }
    
    //Type接口提供了一系列方法
    type Type interface{
    	Align()  int                      //对齐边界
    	FieldAlign()  int
    	Method(int) Method
    	MethodByName(string)  (Method, bool)       //方法
    	NumMethod()  int         //类型名称
    	Name()  string
    	PkgPath()  string         //包路径
    	Slize()  uintptr
    	String()  string
    	Kind()  Kind
    	Implements(u Type)  bool         //是否实现指定接口
    	AssginableTo(u Type)  bool
    	ConvertibleTo(u Type)  bool
    	Comparable()  bool                //是否可比较
    }
    
    在这里插入图片描述

2.反射如何修改变量值

  • 用反射修改变量值,需要用到reflect.Value类型了,如下所示结构体,有三个元素,type字段存储反射变量的类型元数据指针;ptr字段存储数据地址;flag字段是一个位标识符,存储反射变量的描述信息,例如是否为指针,是否为方法或是否只读等等。
    type Value struct {
          type *rtype
          ptr unsafe.Pointer
          flag
    }
    
    func ValueOf(i interface{}) Value{
          if i == nill{
               return Value()
          }
          escapes(i)
          return unpackEface(i)
    }
    
  • 通常会有reflect.ValueOf方法来拿到反射类型的值,注意这里的参数i和TypeOf函数一样是个空接口类型,所以处理方式一样。(编译阶段会增加一个临时变量作为拷贝)
    在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

50 Projects 50 Days - Hidden Search Widget 学习记录

项目地址 Hidden Search Widget 展示效果 Hidden Search Widget 实现思路 点击搜索按钮,展开输入框,主要元素就两个:input输入框和button,这两个本身就是行内元素。点击触发的动作拆分为两个,第一个是input输入框…

Vue核心 事件处理

1.8. 事件处理 1.8.1.事件的基本使用: 使用v-on:xxx或**xxx**绑定事件,其中 xxx 是事件名事件的回调需要配置在methods对象中,最终会在vm上methods中配置的函数,不要用箭头函数,否则this就不是vm了methods中配置的函数&#xff…

手撕Twitter推荐算法

Twitter近期开源了其推荐系统源码[1,2,3],截止现在已经接近36k star。但网上公开的文章都是blog[1]直译,很拗口,因此特地开个系列系统分享下。系列涵盖: Twitter整体推荐系统架构:涵盖图数据挖掘、召回、精排、规则多…

ActiveMQ使用

一、什么是消息中间件 消息中间件顾名思义实现的就是在两个系统或两个客户端之间进行消息传送 二、什么是ActiveMQ ActiveMQ是一种开源的基于JMS(Java Message Servie)规范的一种消息中间件的实现,ActiveMQ的设计目标是提供标准的&#xff0c…

HCIP之VLAN

目录 网络的三层架构 接入层 无线的缺陷: 上网用户数量增多,网络卡顿的原因 CSMA/CD --- 载波侦听多路访问/冲突检测 CSMA/CA --- 载波侦听多路访问/冲突避免 无线网络没有使用冲突检测技术的原因 汇聚层 连接两条线路的原因 核心层 VLAN VLAN配…

数字设计笔试Verilog手撕代码 - 无符号浮点加法器

前言 今天在网上看笔试题发现有个设计浮点累加器的题目,看了下题目说明感觉不太清楚,恰好记得之前做过浮点数的加法运算的设计,索性就改了下题目需求,作为一个小练习在重新设计一遍。具体设计要求如下: 设计需求 设…

Java锁策略-Java多线程(4)

(各位观众老爷下午好, 创作不易勒, 大家多多点赞收藏😘) 咱们废话不多讲, 下面细🔒 目录 前言 乐观锁 VS 悲观锁 乐观锁 悲观锁 悲观乐观锁优缺点 轻量级锁 VS 重量级锁 重量级锁 轻量级锁 读写锁 自旋锁 互斥锁 可重入锁 VS 不可重入锁 死…

IPSCE

文章目录 1.什么是数据认证,有什么作用,有哪些实现的技术手段?2.什么是身份认证,有什么作用,有哪些实现的技术手段?3.什么VPN技术?4.VPN技术有哪些分类?5.IPSEC技术能够提供哪些安全服务?6.IPSEC的技术架构是什么?7.AH与ESP…

今天面了个字阿里拿38K出来的,真是纹身师闭眼,秀了我一脸啊

公司前段缺人,也面了不少测试,前面一开始瞄准的就是中级的水准,也没指望来大牛,提供的薪资在15-20k,面试的人很多,但平均水平很让人失望。看简历很多都是4年工作经验,但面试中,不提测…

8.java程序员必知必会类库之嵌入式SQL数据库

前言 嵌入式内存数据库,作为嵌入到应用内部的数据库,在正常生产业务流程中使用不多。现在一般公司通用架构都是应用和数据分离,解耦数据和应用。但是,在某些特殊场景中,这种嵌入式数据库是比较好的选择。 在某些单元…

如何在开发阶段保证软件工程质量 (程序员要做些什么)

前言 大家常说:“代码和人,有一个能跑就行”,但这并不意味着我们可以放弃职业道德。与土木工程一样,软件工程也需要一些可度量的指标来衡量产品的交付质量。一个高质量的软件绝对不能只靠测试人员来保证,更不能相信程…

在SaleSmartly(ss客服)中为Messenger 提供无缝支持体验

客户希望您在他们所在的地方与他们见面,这意味着打开多个沟通渠道。但是,当您通过电子邮件、实时聊天、社交等方式进行通信时,对话很容易丢失、被忽视和杂乱无章。 而Messenger的受欢迎程度,以及Meta的无所不在,使Face…

每日学术速递4.18

CV - 计算机视觉 | ML - 机器学习 | RL - 强化学习 | NLP 自然语言处理 Subjects: cs.CV 1.Inpaint Anything: Segment Anything Meets Image Inpainting 标题:Inpaint Anything:分割任何东西满足图像修复 作者:Tao Yu, Runseng Feng, R…

企业用户如何选择合适的服务器配置方法教程

随着互联网信息的飞速发展,任何企业都脱离不了互联网,越来越多的企业都通过互联网实施无纸化的办公,互联网推广一体化整体型推广、互联网电子商务。中小型企业网站如何选购云服务器配置呢?但是,实现这些的最最基础的条…

ai改写句子软件-ai改写

AI免费伪原创:助力网站内容升级 您是否曾经为网站优化而烦恼,无论是内容更新还是SEO优化,都需要大量的时间和精力。但是,您是否知道,现在有一款能够使用AI技术来帮助您完成这些任务,而且还是免费的呢&…

【Git 学习】

Git 学习 一、Git的使用1. Git下载安装2. Git 命令3. Git推送代码步骤4. Git基本工作流程5. Git历史版本切换6. Git分支管理6.1 创建新分支6.2 切换分支6.3 合并分支6.4 删除分支 7. 远程仓库的工作流程7.1 具体流程 8.推送到远程仓库9. 代码冲突问题10. IDEA 集成Git10.1 版本…

说说webpack的构建流程?

① 初始化流程 从配置文件和 Shell 语句中读取与合并参数,并初始化需要使用的插件和配置插件等执行环境所需要的参数。 配置文件默认下为 webpack.config.js,也可以通过命令的形式指定配置文件; 主要作用是用于激活webpack的加载项和插件&am…

手写axios源码系列一:axios核心知识点

文章目录 axios的核心功能1、axios 函数对象2、dispatchRequest 发送请求3、interceptors 拦截器4、cancelToken 取消请求 最近从头搭建了一个vue小项目,想使用 axios 作为请求接口的第三方库。结果使用了 axios 这么长时间,想封装一下 axios &#xff0…

Nacos2.2.2开启鉴权配置

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 前言一、更改application.properties中的配置二、修改配置 前言 最近公司开启了一个新的电商项目,项目中用到了Naocs作为注册中心和配置中心&#xff0…

WindowsUbuntu下python程序打包

Python程序的运行必须要有Python的环境,但是程序编出来是用的,如果是给别人用,而他/她的电脑上又没有Python程序运行的环境怎么办呢?总不能让他/她去安装一个吧?这时我们就要将Python程序打包为exe可执行文件&#xff…