【魅力golang】之-反射
1、引言
反射(Reflection)在 Golang中用于运行时检查和操作变量的类型和值。通过反射,可以实现动态类型处理,这在构建泛型代码、框架、序列化工具和动态代理等场景中非常有用。
2、什么是反射
反射是指程序在运行时能够动态地检查变量的类型信息(如类型名、字段、方法等)以及修改变量的值。Go 提供了一整套反射机制,通过内置的 reflect 包支持动态操作。
为什么需要反射
- 动态性:Go 是一种强类型语言,变量类型在编译时确定。反射允许在运行时操作变量的类型和值,提供动态行为。
- 框架设计:许多框架(如 ORM、Web 框架)需要在运行时解析结构体和方法,并进行动态调用。
- 通用处理:在处理未知类型的数据时,反射提供了灵活性,如序列化和反序列化、依赖注入等。
反射的核心设计理念
- 类型(Type)与值(Value)分离:Go 的反射通过 reflect.Type 和 reflect.Value 两个核心类型分别管理类型信息和运行时的值。
- 接口驱动:反射基于接口工作,必须从接口值开始操作。
3、Go 反射的核心概念与用法
反射的核心功能依赖 reflect 包,主要包括以下几个重要概念:
3.1 reflect.Type
reflect.Type 表示变量的类型,用于获取变量的类型信息。
示例:获取类型信息
package mainimport ("fmt""reflect"
)func main() {var x int = 42 // 声明一个变量 x,类型为 intt := reflect.TypeOf(x) // 通过反射 获取变量 x 的类型fmt.Println("Type:", t.Name()) // 输出: intfmt.Println("Kind:", t.Kind()) // 输出: int
}
- Name:获取类型名。
- Kind:获取底层种类(支持结构体、切片、指针等)。
3.2 reflect.Value
reflect.Value 表示变量的值,用于动态获取和修改变量的值。
示例:获取和修改值
package mainimport ("fmt""reflect"
)func main() {var x int = 42 // 示例值v := reflect.ValueOf(x) // 通过反射,获取变量的值fmt.Println("Value:", v.Int()) // 输出: 42// 修改值ptr := reflect.ValueOf(&x) // 获取指针elem := ptr.Elem() // 解引用elem.SetInt(100) // 修改值fmt.Println("Modified Value:", x) // 输出: 100
}
3.3 reflect.Kind
Kind 表示变量的基础种类,如 Struct、Slice、Map、Pointer 等。
示例:区分类型和种类
package mainimport ("fmt""reflect"
)func main() {var x []int // 整型的空切片t := reflect.TypeOf(x) // 通过反射获取切片的类型fmt.Println("Type:", t.Name()) // 输出: 空字符串,因为切片没有名称fmt.Println("Kind:", t.Kind()) // 输出: slice
}
输出:
Type:
Kind: slice
3.4 获取结构体信息
通过反射可以动态获取结构体字段、方法等信息。
示例:获取结构体字段信息
package mainimport ("fmt""reflect"
)type User struct {ID intName string
}func main() {user := User{ID: 1, Name: "Alice"} // 创建一个User结构体实例t := reflect.TypeOf(user) // 获取user结构体的反射类型对象for i := 0; i < t.NumField(); i++ { // 遍历结构体的字段field := t.Field(i) // 获取当前字段的反射类型对象fmt.Printf("Field Name: %s, Type: %s\n", field.Name, field.Type)}
}
输出:
Field Name: ID, Type: int
Field Name: Name, Type: string
4、反射的应用场景
4.1 动态调用方法
反射支持在运行时动态调用方法,适用于插件框架或动态执行的场景。
示例:调用结构体方法
package mainimport ("fmt""reflect"
)type Calculator struct{} // 定义一个结构体func (c Calculator) Add(a, b int) int { // 为这个结构体定义一个方法return a + b
}func main() {calc := Calculator{} // 创建一个Calculator实例v := reflect.ValueOf(calc) // 获取Calculator实例的反射值method := v.MethodByName("Add") // 通过名称获取Calculator实例的方法args := []reflect.Value{reflect.ValueOf(10), reflect.ValueOf(20)} // 创建一个参数列表result := method.Call(args) // 调用方法fmt.Println("Result:", result[0].Int()) // 输出: 30
}
4.2 动态序列化与反序列化
反射常用于实现 JSON、XML 等序列化框架,动态处理不同类型的数据。
示例:JSON 动态序列化
package mainimport ("encoding/json""fmt""reflect"
)func toJSON(data interface{}) string { // 定义一个函数,接收一个interface{}类型的参数,返回一个string类型v := reflect.ValueOf(data) // 获取data的reflect.Value类型if v.Kind() == reflect.Struct { // 判断data是否为结构体类型jsonData, _ := json.Marshal(data) // 将data转换为JSON格式return string(jsonData) // 返回JSON格式的字符串}return ""
}type User struct { // 定义一个用户结构体类型ID intName string
}func main() {user := User{ID: 1, Name: "Alice"} // 创建一个User结构体实例jsonStr := toJSON(user) // 调用toJSON函数,传入user结构体实例fmt.Println("JSON:", jsonStr) // 输出: {"ID":1,"Name":"Alice"}
}
4.3 数据校验
反射可用于动态校验结构体字段。
示例:验证必填字段
package mainimport ("fmt""reflect"
)type User struct { // 定义一个用户结构体Name string `validate:"required"`Age int
}func validateStruct(s interface{}) { // 定义一个验证函数t := reflect.TypeOf(s) // 通过反射获取结构体的类型v := reflect.ValueOf(s) // 通过反射获取结构体的值for i := 0; i < t.NumField(); i++ { // 遍历结构体的每个字段field := t.Field(i) // 获取当前字段的类型tag := field.Tag.Get("validate") // 获取当前字段的标签if tag == "required" && v.Field(i).Interface() == "" { // 如果标签为required且值为空,则输出错误信息fmt.Printf("Field %s is required\n", field.Name) // 输出错误信息}}
}func main() {user := User{} // 创建一个用户结构体实例validateStruct(user) // 输出: Field Name is required
}
5、反射的特点
5.1 特点
- 强大:支持动态检查和操作类型和值。
- 灵活:适用于动态框架、序列化、动态代理等场景。
- 复杂性:代码可读性较低,容易引发错误。
5.2 注意事项
- 性能开销:反射比直接操作慢,频繁使用可能影响性能。
- 类型安全性:反射使用时缺乏类型检查,容易引发运行时错误。
- 接口值限制:反射只能操作接口值,必须通过显式转换或传递接口。
示例:反射的运行时错误
package mainimport ("reflect"
)func main() {var x int = 42v := reflect.ValueOf(x) // v是int类型的反射值v.SetInt(100) // 运行时错误: reflect.Value.SetInt using unaddressable value
}
这里会抛出异常:
panic: reflect: reflect.Value.SetInt using unaddressable value
解决方案:使用指针传递。
6、总结
反射功能强大且复杂,适合在动态类型处理、框架设计等场景中使用。虽然反射提供了极大的灵活性,但也伴随性能开销和复杂性。因此,在实际开发中,应根据需求谨慎使用反射,优先选择静态代码来实现功能。
相关文章:

【魅力golang】之-反射
1、引言 反射(Reflection)在 Golang中用于运行时检查和操作变量的类型和值。通过反射,可以实现动态类型处理,这在构建泛型代码、框架、序列化工具和动态代理等场景中非常有用。 2、什么是反射 反射是指程序在运行时能够动态地检…...
git--批量修改本地用户名和邮箱
原文网址:git--批量修改本地用户名和邮箱-CSDN博客 简介 本文介绍git如何批量修改项目的本地用户名和邮箱。 脚本 新建脚本:git_config.sh,内容如下: #!/bin/bash topDirpwd echo "开始处理" for file in ls ./ do…...

Rofin罗芬激光PowerLine L300 PL400 Manual 软件
Rofin罗芬激光PowerLine L300 PL400 Manual 软件...

【 thefuck 安装与使用】Linux 终端自动纠错工具:一头GitHub上的“草泥马“ - thefuck,妈妈再也不用担心我打错命令行了!
目录 快速安装使用 . 1.简介 2.安装 3.配置 4.补充 官方盗料参考 快速安装使用 快速安装使用,四步即可: #Ubuntu/Debian系统 sudo apt update sudo apt install python3-dev python3-pip sudo pip3 install thefuck #编辑bashrc配置文件 vim ~/.bashrc…...

牛客网刷题 ——C语言初阶——BC112小乐乐求和
1.牛客网刷题 ——C语言初阶 牛客网:BC112小乐乐求和 小乐乐最近接触了求和符号Σ,他想计算的结果。但是小乐乐很笨,请你帮助他解答。 输入描述: 输入一个正整数n (1 ≤ n ≤ 109) 输出描述: 输出一个值,为求和结果。 示例1 输…...

【PyTorch】(基础七)---- 完整训练流程
首先要明确一点,我们在编写模型、训练和使用模型的时候通常都是分开的,所以应该把Module的编写以及train方法和test方法分开编写。 调用gpu进行训练:在网络模型,数据,损失函数对象后面都使用.cuda(&#x…...

01- 三自由度串联机械臂位置分析
三自由度串联机械臂如下图所示(d180mm,L1100mm,L280mm),利用改进DH法建模,坐标系如下所示: 利用改进DH法建模,该机器人的DH参数表如下所示: 对该机械臂进行位置分析&…...

Flutter实现可拖拽操作Draggable
文章目录 1. Draggable 控件的构造函数主要参数: 2. Draggable 的工作原理3. 常见用法示例 1:基本的拖拽控件解释:示例 2:与 DragTarget 配合使用解释: 4. Draggable 的回调详解5. 总结 Draggable 是 Flutter 中一个用…...

Vue BPMN Modeler流程图
1、参考地址 git clone https://github.com/evanyangg/vue-bpmn-modeler.git 2、安装bpmn.js npm install bpmn-js --save 3、使用bpmn.js <template><div class"containers"><div class"canvas" ref"canvas"></div&g…...
写在公司40周年前夕
日子太快了,来这里工作六年多了。现在才知道原来入职的公司只是母公司的一小点。刚来一年就碰到疫情,三年疫情之后就迎来亏损,而后就是变了董事长,换了总经理。 这圣诞前,所有的子分又换了一把手。动作之大,…...
Python调用Elasticsearch更新数据库
文章目录 Elasticsearch介绍Python调用Elasticsearch更新数据库 Elasticsearch介绍 Elasticsearch是一个基于Lucene的搜索引擎,它提供了一个分布式、多租户能力的全文搜索引擎,具有HTTP web接口和无模式的JSON文档。Elasticsearch是用Java开发的&#x…...

测试基础之测试分类
软件测试是确保软件产品满足预期功能、性能和用户体验要求的关键环节。它的主要目的是通过系统化的方法发现并修复软件中的缺陷,从而提高软件的质量和可靠性。在软件开发生命周期的不同阶段执行测试,以尽早发现潜在的错误或类型,早期发现缺陷…...

太阳能LED路灯智能控制系统(论文+源码)
1系统的功能及方案设计 本次课题为太阳能LED路灯智能控制系统,其系统整体架构如图2.1所示,太阳能板通过TP4056充电模块给锂电池进行充电,电池通过HX3001升压模块进行升压到5V给整个控制系统进行供电,控制系统由AT89C52单片机作为…...
文本数据处理
文本数据处理 一、数据转换与错误处理 (一)运维中的数据转换问题 在计算机审计及各类数据处理场景中,数据转换是关键步骤,涉及将被审计单位或其他来源的数据有效装载到目标数据库,并明确标示各表及字段含义与关系。…...

Liunx环境下安装人大金仓数据库V8R6版本
Liunx环境下安装人大金仓数据库V8R6版本 一:硬件环境要求二:软件环境要求三:安装包准备四:检测和配置环境4.1:检查操作系统信息4.2 检查系统内存与存储空间 五:配置内核参数六:预安装工作6.1 创…...
Android使用PorterDuffXfermode模式PorterDuff.Mode.SRC_OUT橡皮擦实现马赛克效果,Kotlin(3)
Android使用PorterDuffXfermode模式PorterDuff.Mode.SRC_OUT橡皮擦实现马赛克效果,Kotlin(3) import android.content.Context import android.graphics.Bitmap import android.graphics.BitmapFactory import android.graphics.Canvas impor…...

python 怎么引入类
一、导入单个类 from fun import Dog dogDog(husike) dog.bark() 二、导入多个类 多个类之间用逗号分隔 from fun import Dog,Cat dogDog(husike) dog.bark() catCat(maomi) cat.catch_mouse() 三、导入整个模块 import fun dogfun.Dog(husike) dog.bark() catfun.Cat(maomi) …...

Day35汉明距离
两个整数之间的 汉明距离 指的是这两个数字对应二进制位不同的位置的数目。 给你两个整数 x 和 y,计算并返回它们之间的汉明距离。 class Solution {public int hammingDistance(int x, int y) {int cnt 0;while (Math.max(x, y) ! 0) {if ((x & 1) ! (y &…...

中文学习系统:客户服务与学习支持
3.1 系统可行性分析 开发一款程序软件不仅需要时间,也需要人力,物力资源。而进行可行性分析这个环节就是解决用户这方面的疑问,看看程序在当前的条件下是否可以进行开发。 3.1.1 技术可行性分析 此程序选用的开发语言是Java,这种编…...

华为麦芒5(安卓6)termux记录 使用ddns-go,alist
下载0.119bate1版,不能换源,其他源似乎都用不了,如果root可以直接用面具模块 https://github.com/termux/termux-app/releases/download/v0.119.0-beta.1/termux-app_v0.119.0-beta.1apt-android-5-github-debug_arm64-v8a.apk 安装ssh(非必要) pkg install openssh开启ssh …...
RestClient
什么是RestClient RestClient 是 Elasticsearch 官方提供的 Java 低级 REST 客户端,它允许HTTP与Elasticsearch 集群通信,而无需处理 JSON 序列化/反序列化等底层细节。它是 Elasticsearch Java API 客户端的基础。 RestClient 主要特点 轻量级ÿ…...
从零实现富文本编辑器#5-编辑器选区模型的状态结构表达
先前我们总结了浏览器选区模型的交互策略,并且实现了基本的选区操作,还调研了自绘选区的实现。那么相对的,我们还需要设计编辑器的选区表达,也可以称为模型选区。编辑器中应用变更时的操作范围,就是以模型选区为基准来…...
在rocky linux 9.5上在线安装 docker
前面是指南,后面是日志 sudo dnf config-manager --add-repo https://download.docker.com/linux/centos/docker-ce.repo sudo dnf install docker-ce docker-ce-cli containerd.io -y docker version sudo systemctl start docker sudo systemctl status docker …...
css的定位(position)详解:相对定位 绝对定位 固定定位
在 CSS 中,元素的定位通过 position 属性控制,共有 5 种定位模式:static(静态定位)、relative(相对定位)、absolute(绝对定位)、fixed(固定定位)和…...
Matlab | matlab常用命令总结
常用命令 一、 基础操作与环境二、 矩阵与数组操作(核心)三、 绘图与可视化四、 编程与控制流五、 符号计算 (Symbolic Math Toolbox)六、 文件与数据 I/O七、 常用函数类别重要提示这是一份 MATLAB 常用命令和功能的总结,涵盖了基础操作、矩阵运算、绘图、编程和文件处理等…...

基于Springboot+Vue的办公管理系统
角色: 管理员、员工 技术: 后端: SpringBoot, Vue2, MySQL, Mybatis-Plus 前端: Vue2, Element-UI, Axios, Echarts, Vue-Router 核心功能: 该办公管理系统是一个综合性的企业内部管理平台,旨在提升企业运营效率和员工管理水…...

[论文阅读]TrustRAG: Enhancing Robustness and Trustworthiness in RAG
TrustRAG: Enhancing Robustness and Trustworthiness in RAG [2501.00879] TrustRAG: Enhancing Robustness and Trustworthiness in Retrieval-Augmented Generation 代码:HuichiZhou/TrustRAG: Code for "TrustRAG: Enhancing Robustness and Trustworthin…...

Visual Studio Code 扩展
Visual Studio Code 扩展 change-case 大小写转换EmmyLua for VSCode 调试插件Bookmarks 书签 change-case 大小写转换 https://marketplace.visualstudio.com/items?itemNamewmaurer.change-case 选中单词后,命令 changeCase.commands 可预览转换效果 EmmyLua…...
前端高频面试题2:浏览器/计算机网络
本专栏相关链接 前端高频面试题1:HTML/CSS 前端高频面试题2:浏览器/计算机网络 前端高频面试题3:JavaScript 1.什么是强缓存、协商缓存? 强缓存: 当浏览器请求资源时,首先检查本地缓存是否命中。如果命…...

车载诊断架构 --- ZEVonUDS(J1979-3)简介第一篇
我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 做到欲望极简,了解自己的真实欲望,不受外在潮流的影响,不盲从,不跟风。把自己的精力全部用在自己。一是去掉多余,凡事找规律,基础是诚信;二是…...