go的反射

1.介绍

        有时我们需要写一个函数,这个函数有能力统一处理各种值类型,而这些类型可能无法共享同一个接口,也可能布局未知,也有可能这个类型在我们设计函数时还不存在,这个时候我们就可以用到反射。

        空接口可以存储任意类型的变量,那我们如何知道这个空接口保存数据的类型是什么? 值是什么呢?

  •         可以使用类型断言
  •         可以使用反射实现,也就是在程序运行时动态的获取一个变量的类型信息和值信息。

         把结构体序列化成json字符串,自定义结构体Tab标签的时候就用到了反射,ORM框架,底层就是用到了反射技术。

        ORM:对象关系映射(Object Relational Mapping,简称 ORM)是通过使用描述对象和数据库之间的映射的元数据,将面向对象语言程序中的对象自动持久化到关系数据库中。

2. 反射的定义

        反射是指在程序运行期间对程序本身进行访问和修改的能力。正常情况程序在编译时,变量被转换为内存地址,变量名不会被编译器写入到可执行部分。在运行程序时,程序无法获取自身的信息。支持反射的语言可以在程序编译期将变量的反射信息,如字段名称、类型信息、结构体信息等整合到可执行文件中,并给程序提供接口访问反射信息,这样就可以在程序运行期获取类型的反射信息,并且有能力修改它们。

3.反射可以实现的功能

  • 反射可以在程序运行期间动态的获取变量的各种信息,比如变量的类型类别
  • 如果是结构体,通过反射还可以获取结构体本身的信息,比如结构体的字段、结构体的方法。
  • 通过反射,可以修改变量的值,可以调用关联的方法

Go语言中的变量是分为两部分的:

  • 类型信息:预先定义好的元信息。
  • 值信息:程序运行过程中可动态变化的。

        在Go语言的反射机制中,任何接口值都由是一个具体类型和具体类型的值两部分组成的。

        在Go语言中反射的相关功能由内置的reflect包提供,任意接口值在反射中都可以理解为由 reflect.Type 和 reflect.Value两部分组成,并且reflect包提供了reflect.TypeOf和reflect.ValueOf两个重要函数来获取任意对象的Value 和 Type

4. reflect.TypeOf()获取任意值的类型

        在Go 语言中,使用reflect.TypeOf()函数可以接受任意类型参数,可以获得任意值的类型对象(reflect.Type),程序通过类型对象可以访问任意值的类型信息。

func getReflectInfo(x interface{}) {v := reflect.TypeOf(x)fmt.Println(v)
}func main() {getReflectInfo(10)getReflectInfo(10.01)getReflectInfo("abc")getReflectInfo(true)
}

5. type name 和type kind

        在反射中关于类型还划分为两种:类型(Type)和种类(Kind)。因为在Go语言中我们可以使用type关键字构造很多自定义类型,而种类(Kid)就是指底层的类型,但在反射中,当需要区分指针、结构体等大品种的类型时,就会用到种类(Kind)。举个例子,我们定义了两个指针类型和两个结构体类型,通过反射查看它们的类型和种类。

        Go 语言的反射中像数组、切片、Map、指针等类型的变量,它们的.Name()都是返回空。

func getReflectInfo(x interface{}) {v := reflect.TypeOf(x)fmt.Println("类型 ", v)fmt.Println("类型名称 ", v.Name())fmt.Println("类型种类 ", v.Kind())
}

使用断言和反射实现空接口类型的数相加

断言式:

func getReflectInfo(x interface{}) {b, _ := x.(int)var num = 10 + bfmt.Println(num)
}

反射式:

func getReflectInfo(x interface{}) {v := reflect.ValueOf(x)var n = v.Int() + 10fmt.Println(n)
}

使用swith判断类型

func getReflectInfo(x interface{}) {v := reflect.TypeOf(x)kind := v.Kind()switch kind {case reflect.Int:fmt.Println("我是int类型")case reflect.Float64:fmt.Println("我是float64类型")default:fmt.Println("我是其它类型")}
}

6 reflect.ValueOf

reflect.ValueOf() 返回的是reflect.Value类型,其中包含了原始值的值信息,reflect.Value与原始值之间可以互相转换

reflect.value类型提供的获取原始值的方法如下

方法说明
interface()将值以interface{}类型返回,可以通过类型断言转换为指定类型
Int() int64将值以int类型返回,所有有符号整型均可以此方式返回
Uint() uint64将值以uint类型返回,所有无符号整型均可以以此方式返回
Float() float64将值以双精度(float 64)类型返回,所有浮点数(float 32、float64)均可以以此方式返回

7. 结构体的反射

任意值通过reflect.Typeof)获得反射对象信息后,如果它的类型是结构体,可以通过反射值对象(reflect.Type)的NumField()和Field()方法获得结构体成员的详细信息。

reflect.Type中与获取结构体成员相关的的方法如下表所示。

方法说明
Field(i int)StructField根据索引,返回索引对应的结构体字段的信息
NumField() int返回结构体成员字段数量
FieldByName(name string)(StructField, bool)根据给定字符串返回字符串赌赢的结构体字段信息
FieldByIndex(index []int)StructField多层成员访问时,根据[] int 提供的每个结构

结构体中获取属性操作

//结构体
type Student struct {Name  string `json: "name"`Age   int    `json: "age"`Score int    `json: "score"`
}
func PrintStructField(s interface{}) {t := reflect.TypeOf(s)// 判断传递过来的是否是结构体if t.Kind() != reflect.Struct && t.Elem().Kind() != reflect.Struct {fmt.Println("这个不是结构体")return}// 通过类型变量里面的Field可以获取结构体的字段field0 := t.Field(0) // 获取第0个字段fmt.Printf("%#v \n", field0)fmt.Println("字段名称:", field0.Name)fmt.Println("字段类型:", field0.Type)fmt.Println("字段Tag:", field0.Tag.Get("json"))// 通过类型变量里面的FieldByName可以获取结构体的字段中field1, ok := t.FieldByName("Age")if ok {fmt.Println("字段名称:", field1.Name)fmt.Println("字段类型:", field1.Type)fmt.Println("字段Tag:", field1.Tag)}// 通过类型变量里面的NumField获取该结构体有几个字段var fieldCount = t.NumField()fmt.Println("结构体有:", fieldCount, " 个属性")// 获取结构体属性对应的值v := reflect.ValueOf(s)nameValue := v.FieldByName("Name")fmt.Println("nameValue:", nameValue)}

对方法的操作

type Student struct {Name  string `json:"name"`Age   int    `json:"age"`Score int    `json:"score"`
}func (s Student) GetInfo() string {var str = fmt.Sprintf("姓名:%v 年龄:%v 成绩:%v", s.Name, s.Age, s.Score)return str
}func (s *Student) SetInfo(name string, age int, score int) {s.Name = names.Age = ages.Score = score
}func (s Student) PrintStudent() {fmt.Println(s.GetInfo())
}func getFun(x interface{}) {v := reflect.ValueOf(x)t := v.Type()// 检查x是否是指向结构体的指针if t.Kind() != reflect.Ptr || t.Elem().Kind() != reflect.Struct {return}// 获取指针指向的结构体的反射值,并使其可变elem := v.Elem()// 打印结构体拥有的方法名和方法数量methodCount := t.Elem().NumMethod()for i := 0; i < methodCount; i++ {method := t.Elem().Method(i)fmt.Println("方法名:", method.Name)}// 调用PrintStudent方法elem.MethodByName("PrintStudent").Call(nil)// 准备参数并调用SetInfo方法var params = []reflect.Value{reflect.ValueOf("李四"),reflect.ValueOf(23),reflect.ValueOf(98),}v.MethodByName("SetInfo").Call(params)// 再次调用PrintStudent方法elem.MethodByName("PrintStudent").Call(nil)
}func main() {student := &Student{"张三", 18, 95} // 创建一个指向Student的指针getFun(student)
}

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

WPF+MVVM案例实战(十八)- 自定义字体图标按钮的封装与实现(ABD类)

文章目录 1、案例效果1、按钮分类2、ABD类按钮实现描述1.文件创建与代码实现2、样式引用与控件封装3、按钮案例演示1、页面实现与文件创建2、运行效果如下3、总结4、源代码获取1、案例效果 1、按钮分类 在WPF开发中,最常见的就是按钮的使用,这里我们总结以下大概的按钮种类,…

ARM base instruction -- mneg

Multiply-Negate multiplies two register values, negates the product, and writes the result to the destination register. 乘法-求反&#xff0c;将两个寄存器值相乘&#xff0c;对乘积求反&#xff0c;并将结果写入目标寄存器。 32-bit variant Applies when sf 0…

【鸿蒙新闻】10月29日警用鸿蒙开发者大会在北京胜利召开,开启智慧应用新时代!

10月29日&#xff0c;在公安部科技信息化局、公安部装备财务局指导下&#xff0c;由公安部第一研究所主办&#xff0c;鼎桥通信技术有限公司、OpenHarmony生态委员会及公共安全专委会协办的警用鸿蒙开发者大会在北京胜利召开。会议以“拥抱警鸿创新生态 开启智慧应用新时代”为…

架构师备考-软件工程相关补充

软件开发生命周期 按照传统的软件生命周期方法学&#xff0c;可以把软件生命周期划分为软件定义、软件开发、软件运行与维护三个阶段。 软件定义&#xff1a;软件定义包括可行性研究和详细需求分析过程&#xff0c;任务是确定软件开发工程必须完成的目标。具体可分为问题定义、…

OpenGL入门003——使用Factory设计模式简化渲染流程

前面两节已经学会了如何使用opengl创建窗口并绘制三角形&#xff0c;我们可以看出有些步骤是固定的&#xff0c;而且都写在main.cpp&#xff0c;这一节我们将了解如何使用Factroy设计模型。将模型渲染逻辑封装在一个单独的类中&#xff0c;简化开发流程&#xff0c;且提高代码复…

音频中sample rate是什么意思?

‌sample rate‌在数字信号处理中&#xff0c;指的是‌采样频率‌&#xff0c;即每秒钟从连续信号中抽取的样本数量。采样频率越高&#xff0c;信号的还原度越高&#xff0c;但同时也会增加计算负担和存储需求‌。 实际应用场景 在音频处理中&#xff0c;设置合适的采样率可以…

分享一下面试中常用的10 个面试点全解析,面试成功的秘诀

大家好&#xff0c;我是一颗甜苞谷&#xff0c;今天分享一下面试中常用的10 个面试点全解析,助你面试中脱颖而出 问题1&#xff1a;微服务架构和传统架构有什么区别&#xff0c;现在市场上的微服务架构有哪些? 答&#xff1a;传统的单体架构可维护性、可读性低&#xff0c;维…

构建品牌影响力:知识库工具在市场营销中的创新应用

在当今这个信息爆炸的时代&#xff0c;品牌影响力成为了企业市场竞争力的核心要素。为了有效提升品牌影响力&#xff0c;企业不仅需要精准的市场定位和优质的产品服务&#xff0c;还需要借助高效、智能的知识库工具来优化其市场营销策略。本文将探讨知识库工具在市场营销中的创…

Python Matplotlib 子图绘制

Python 中的子图绘制 在数据可视化中&#xff0c;展示多个图表在同一个画布上是常见的需求&#xff0c;这样可以更直观地比较不同数据集之间的关系。Python 中的 Matplotlib 库为我们提供了强大的功能来实现这一点。在本篇文章中&#xff0c;我们将详细介绍如何使用 Matplotli…

探索设计模式:命令模式

探索设计模式&#xff1a;命令模式 &#x1f9d0;1. 概念&#x1f3af;2. 作用&#x1f4e6;3. 实现3.1 定义命令接口3.2 实现具体命令3.3 实现接收者3.4 实现调用者3.5 使用 &#x1f4bb;4. 应用场景 命令模式&#xff08;Command Pattern&#xff09;就是一种行为型设计模式…

Python-创建并调用自定义文件中的模块/函数

背景&#xff1a;在Python编程中&#xff0c;我们常常需要创建自己的专属文件&#xff0c;以便帮助我们更高效&#xff0c;快捷地完成任务。那么在Python中我们怎么创建并调用自己文件中的模块/函数呢? 在Python中调用自定义文件&#xff0c;通常是指调用自己编写的Python模块…

springboot 修复 Spring Framework 特定条件下目录遍历漏洞(CVE-2024-38819)

刚解决Spring Framework 特定条件下目录遍历漏洞&#xff08;CVE-2024-38816&#xff09;没几天&#xff0c;又来一个新的&#xff0c;真是哭笑不得啊。 springboot 修复 Spring Framework 特定条件下目录遍历漏洞&#xff08;CVE-2024-38816&#xff09;https://blog.csdn.ne…

AutoDIR: Automatic All-in-One Image Restoration with Latent Diffusion论文阅读笔记

AutoDIR: Automatic All-in-One Image Restoration with Latent Diffusion 论文阅读笔记 这是ECCV2024的论文&#xff0c;作者单位是是港中文和上海AI Lab 文章提出了一个叫AutoDIR的方法&#xff0c;包括两个关键阶段&#xff0c;一个是BIQA&#xff0c;基于vision-language…

CDN加速实战:使用七牛云CDN加速阿里云OSS资源访问

今天是双11搞活动,在阿里云1元注册了个域名,想着在学CDN,想使用CDN做个加速项目,但是阿里的要收费,上网查了下七牛云的不收费,想着将七牛云的CDN结合阿里的DNS做个访问加速,刚好看到了阿里的一个文章,照着改了改,实践成功了。 阿里文章:使用CDN加速OSS资源访问_对象…

MacBook 如何设置打开json格式文件的默认程序是vs code

首先右键选中文件&#xff0c;然后选中显示简介 然后选中打开方式 设置成vs code

HTML 基础标签——文本内容标签 <ul>、<ol>、<blockquote> 、<code> 等标签的用法详解

文章目录 1. 标题标签2. 段落标签3. 文本格式化标签4. 列表标签4.1 无序列表 `<ul>`4.2 有序列表 `<ol>`5. 引用标签5.1 块引用 `<blockquote>`5.2 行内引用 `<q>`5.3 作品引用 `<cite>`6. 代码和预格式文本标签6.1 代码标签 `<code>`6.2 …

git 删除远程不存在本地命令却能看到的分支

要删除远程不存在但本地却能看到的分支&#xff0c;你可以按照以下步骤操作&#xff1a; 删除本地分支&#xff1a; 如果你确定要删除的分支已经没有用处&#xff0c;可以使用以下命令来删除本地分支&#xff1a; git branch -d <branch-name>这里的 <branch-name>…

【Oracle APEX开发小技巧10】CSS样式控制交互式报表列宽和自动换行效果

在实际开发中使用交互式报表可能会出现某些字段的列宽过长&#xff0c;某些字段的列宽只有缩到一角的情况&#xff0c;那么如何解决这种情况呢&#xff1f;有没有方法可以控制交互式报表的列宽呢&#xff1f;下面就来介绍一下解决方法&#xff1a; 页设置-页-CSS-内嵌 输入如下…

Linux内核、线程、进程同步互斥方法及IPC方法的总结

前段实践在B站进行模拟面试时发现&#xff0c; 模拟面试第四期-已经拿到大厂OFFER的研究生大佬-LINUX卷到飞起 自己对Linux中的同步互斥方法&#xff0c;以及IPC方法&#xff0c;没有很好的理解和总结过。因此&#xff0c;本笔记将总结这部分内容。 内核线程进程机制原子操作、…

Automated Isotope Identification Algorithm UsingArtificial Neural Networks-论文阅读

Automated Isotope Identification Algorithm Using Artificial Neural Networks 使用人工神经网络的自动同位素识别算法 M.Kamuda, J. Stinnett, and C.J. Sullivan 摘要 需要开发一种算法,以确定包含多种放射性同位素混合物的大量低分辨率伽马射线光谱数据集中放射性同位素…