Go语言反射从入门到进阶
一、反射的基础概念
在 Go 语言中,反射是程序在运行时检查和修改自身状态的能力。通过反射,我们可以在运行时获取变量的类型信息、查看结构体的字段、调用方法等。Go 语言的反射功能主要通过 reflect
包实现。
1.1 反射的基本类型:Type 和 Value
Go 的反射主要基于两个重要的类型:
reflect.Type
:表示 Go 类型的接口reflect.Value
:表示 Go 值的接口
让我们从一个简单的例子开始:
package mainimport ("fmt""reflect"
)func main() {var x float64 = 3.14// 获取变量的类型信息t := reflect.TypeOf(x)fmt.Printf("类型:%v\n", t)// 获取变量的值信息v := reflect.ValueOf(x)fmt.Printf("值:%v\n", v)
}
二、基本类型的反射操作
2.1 获取类型信息
package mainimport ("fmt""reflect"
)func main() {var num int64 = 42var str string = "hello"// 获取基本类型信息fmt.Printf("num 的类型:%v\n", reflect.TypeOf(num))fmt.Printf("str 的类型:%v\n", reflect.TypeOf(str))// 获取类型的种类(Kind)fmt.Printf("num 的种类:%v\n", reflect.TypeOf(num).Kind())fmt.Printf("str 的种类:%v\n", reflect.TypeOf(str).Kind())
}
2.2 获取和修改值
package mainimport ("fmt""reflect"
)func main() {x := 3.14v := reflect.ValueOf(&x) // 注意:这里传入指针// 检查值是否可以被修改if v.Kind() == reflect.Ptr && v.Elem().CanSet() {v.Elem().SetFloat(2.718)}fmt.Printf("修改后的值:%v\n", x)
}
三、结构体的反射操作
3.1 基本结构体反射
package mainimport ("fmt""reflect"
)type Person struct {Name string `json:"name"`Age int `json:"age"`
}func main() {p := Person{Name: "张三",Age: 25,}t := reflect.TypeOf(p)// 遍历结构体字段for i := 0; i < t.NumField(); i++ {field := t.Field(i)fmt.Printf("字段名:%s\n", field.Name)fmt.Printf("字段类型:%v\n", field.Type)fmt.Printf("标签:%v\n", field.Tag.Get("json"))}
}
3.2 动态调用方法
package mainimport ("fmt""reflect"
)type Person struct {Name stringAge int
}func (p Person) SayHello(msg string) string {return fmt.Sprintf("Hello, %s. I am %s", msg, p.Name)
}func main() {p := Person{Name: "张三", Age: 25}// 获取方法v := reflect.ValueOf(p)method := v.MethodByName("SayHello")// 准备参数args := []reflect.Value{reflect.ValueOf("世界")}// 调用方法result := method.Call(args)fmt.Println(result[0].String())
}
四、高级应用场景
4.1 通用的结构体字段验证器
package mainimport ("fmt""reflect""strings"
)type User struct {Name string `validate:"required,min=3"`Email string `validate:"required,email"`Age int `validate:"required,min=18"`
}func validate(v interface{}) []string {var errors []stringt := reflect.TypeOf(v)val := reflect.ValueOf(v)for i := 0; i < t.NumField(); i++ {field := t.Field(i)value := val.Field(i)// 获取验证规则rules := strings.Split(field.Tag.Get("validate"), ",")for _, rule := range rules {switch {case rule == "required":if value.Interface() == reflect.Zero(value.Type()).Interface() {errors = append(errors, fmt.Sprintf("%s 是必填字段", field.Name))}case strings.HasPrefix(rule, "min="):// 这里简化处理,实际应用中需要更复杂的验证逻辑if value.Kind() == reflect.String && len(value.String()) < 3 {errors = append(errors, fmt.Sprintf("%s 长度不能小于3", field.Name))}}}}return errors
}func main() {user := User{Name: "张",Email: "invalid-email",Age: 16,}if errors := validate(user); len(errors) > 0 {fmt.Printf("验证错误:\n%s\n", strings.Join(errors, "\n"))}
}
4.2 通用的 JSON 序列化器
package mainimport ("fmt""reflect""strings"
)func toJSON(v interface{}) string {t := reflect.TypeOf(v)val := reflect.ValueOf(v)if t.Kind() != reflect.Struct {return fmt.Sprintf("\"%v\"", val.Interface())}var pairs []stringfor i := 0; i < t.NumField(); i++ {field := t.Field(i)value := val.Field(i)// 获取json标签jsonTag := field.Tag.Get("json")if jsonTag == "" {jsonTag = field.Name}pair := fmt.Sprintf("\"%s\":%v", jsonTag, toJSON(value.Interface()))pairs = append(pairs, pair)}return "{" + strings.Join(pairs, ",") + "}"
}type Address struct {Street string `json:"street"`City string `json:"city"`
}type Person struct {Name string `json:"name"`Age int `json:"age"`Address Address `json:"address"`
}func main() {p := Person{Name: "张三",Age: 25,Address: Address{Street: "中关村大街",City: "北京",},}fmt.Println(toJSON(p))
}
五、反射的最佳实践
-
谨慎使用反射:
- 反射会带来性能开销
- 代码可读性可能降低
- 编译时类型检查被绕过
-
适合使用反射的场景:
- 需要处理未知类型的数据
- 需要动态调用方法
- 需要实现通用的框架或库
- 需要根据配置动态创建对象
-
性能优化建议:
- 缓存反射结果
- 避免重复获取 Type 和 Value
- 在性能敏感的代码路径上避免使用反射
六、总结
Go 语言的反射机制为我们提供了强大的运行时类型信息和值操作能力。通过反射,我们可以:
- 检查类型信息
- 获取和修改值
- 访问结构体字段和方法
- 实现通用的框架和工具
但是,反射也带来了性能开销和代码复杂性,因此应该在合适的场景下谨慎使用。在实际开发中,建议遵循"实用性"原则,在确实需要反射的场景下再使用它。
相关文章:
Go语言反射从入门到进阶
一、反射的基础概念 在 Go 语言中,反射是程序在运行时检查和修改自身状态的能力。通过反射,我们可以在运行时获取变量的类型信息、查看结构体的字段、调用方法等。Go 语言的反射功能主要通过 reflect 包实现。 1.1 反射的基本类型:Type 和 …...

【基于rust-wasm的前端页面转pdf组件和示例】
基于rust-wasm前端页面转pdf组件和示例 朔源多余的废话花哨的吹牛那点东西要不要拿来试试事到如今 做个美梦 我觉得本文的意义在于,wasm扩展了浏览器的边界,但是又担心如同java的web applet水土不服. 如同我至今看不出塞班和iOS的不同下载地址:在github的备份 朔源…...

ARM64 Windows 10 IoT工控主板运行x86程序效率测试
ARM上的 Windows 10 IoT 企业版支持仿真 x86 应用程序,而 ARM上的 Windows 11 IoT 企业版则支持仿真 x86 和 x64 应用程序。英创推出的名片尺寸ARM64工控主板ESM8400,可预装正版Windows 10 IoT企业版操作系统,x86程序可无需修改而直接在ESM84…...

开放世界目标检测 Grounding DINO
开放世界目标检测 Grounding DINO flyfish Grounding DINO 是一种开创性的开放集对象检测器,它通过结合基于Transformer的检测器DINO与基于文本描述的预训练技术,实现了可以根据人类输入(如类别名称或指代表达)检测任意对象的功…...

easegen将教材批量生成可控ppt课件方案设计
之前客户提出过一个需求,就是希望可以将一本教材,快速的转换为教学ppt,虽然通过人工程序脚本的方式,已经实现了该功能,但是因为没有做到通用,每次都需要修改脚本,无法让客户自行完成所有流程&am…...

2002 - Can‘t connect to server on ‘192.168.1.XX‘ (36)
参考:2002 - Can‘t connect to server on ‘192.168.1.XX‘ (36) ubantu20.04,mysql5.7.13 navicat 远程连接数据库报错 2002 - Can’t connect to server on ‘192.168.1.61’ (36) 一、查看数据库服务是否有启动,发现有启动 systemctl status mysql…...

【虚拟机网络拓扑记录】
虚拟机网络拓扑记录 虚拟机安装windows到ubuntu的网络拓扑ubuntu到ubuntu里面的虚拟机网络拓扑windows到ubuntu里面的虚拟机网络拓扑 虚拟机安装 本实验宿主机为windos, 安装vmware,虚拟机操作系统使用ubuntu,然后再在ubuntu上面创建新的虚拟…...

【单片机通讯协议】—— 常用的UART/I2C/SPI等通讯协议的基本原理与时序分析
提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 前言一、通信基本知识1.1 MCU的参见外设1.2 通信的分类按基本的类型从传输方向上来分 二、UART(串口通讯)2.1 简介2.2 时序图分析2.3 UART的…...

Vue3 核心语法
1. OptionsAPI 与 CompositionAPI Vue2 的API设计是 Options(配置)风格的。Vue3 的API设计是 Composition(组合)风格的。 1.1 Options API 的弊端 Options类型的 API,数据、方法、计算属性等,是分散在&a…...

LLaMA-Factory GLM4-9B-CHAT LoRA 指令微调实战
🤩LLaMA-Factory GLM LoRA 微调 安装llama-factory包 git clone --depth 1 https://github.com/hiyouga/LLaMA-Factory.git进入下载好的llama-factory,安装依赖包 cd LLaMA-Factory pip install -e ".[torch,metrics]" #上面这步操作会完成…...

GTM023 W.H.Greub线性代数经典教材:Linear Algebra
这本教材是我高中时期入门线性代数的主要教材,我的很多基础知识都来源于这本书,如今看回这本书可以说满满的回忆。这本书可以说,是我读过的内容最为全面且完备的线性代数教材了。而且它的语言风格非常的代数化,没有什么直观可言&a…...
交换机与路由器的区别
交换机和路由器是网络中的两种关键设备,它们各自承担不同的功能,主要区别体现在以下几个方面: 一、工作层次与功能 交换机: 工作层次:交换机主要工作在OSI模型的第二层,即数据链路层。 功能:交…...

springboot502基于WEB的牙科诊所管理系统(论文+源码)_kaic
牙科诊所管理系统的设计与实现 摘要 近年来,信息化管理行业的不断兴起,使得人们的日常生活越来越离不开计算机和互联网技术。首先,根据收集到的用户需求分析,对设计系统有一个初步的认识与了解,确定牙科诊所管理系统的…...

soular使用教程
用 soular 配置你的组织,工作更高效!以下是快速上手的简单步骤:  1. 账号管理 可以对账号信息进行多方面管理,包括分配不同的部门、用户组等,从而确保账号权限和职责的清晰分配。  1.1 用…...

纯div+css+js弹出窗
目的:实现弹出窗、仅关闭弹窗之后才能操作。自适应宽度与高度、当文本内容太多时、添加滚动条效果。 效果图 源码 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8" /><meta name"viewport"…...

一篇文章学会HTML
目录 页面结构 网页基本标签 图像标签 超链接标签 文本链接 图像链接 锚链接 功能链接 列表 有序列表 无序列表 自定义列表 表格 跨列/跨行 表头 媒体元素 视频 音频 网站的嵌套 表单 表单元素 文本框 单选框 多选框 按钮 下拉框 文本域和文件域 表…...

QGIS二次开发(插件开发)
实习二 QGIS插件开发 2.1 任务要求 a)用C语言编写qgis插件,实现带有x/y坐标的文本文件的地图显示。 用文件流fstream操作文本文件,读取其中的坐标数据。基于QgsPlugin相关类派生出一个插件,并加到插件工厂中。基于QgsVectorLaye…...
Web防火墙和下一代防火墙的区别
介绍 客户经常询问“当我已经拥有下一代防火墙(NGFW)时,为什么需要Web应用程序防火墙(WAF)?”。本博文的目的是解释两种解决方案之间的区别,重点关注Web应用程序防火墙可以提供的附加值。 什么…...
Linux:alias别名永久有效
一、背景 日常使用bash时候,有些常用的命令参数的组合命令太长,很难记,此时可以利用Linux提供的alias命令生成命令的别名(命令的隐射),但是我们会发现,当退出了终端后重新登录就失效了ÿ…...

【递归与回溯深度解析:经典题解精讲(中篇)】—— LeetCode
文章目录 组合目标和组合总和字母大小写全排序优美的排列N皇后 组合 思路:回溯算法 问题要求从 1 到 n 中选出 k 个数的所有组合。 使用回溯算法递归构造解。 每次递归时,记录当前的组合路径,当组合长度达到 k 时,将其加入结果集…...

idea大量爆红问题解决
问题描述 在学习和工作中,idea是程序员不可缺少的一个工具,但是突然在有些时候就会出现大量爆红的问题,发现无法跳转,无论是关机重启或者是替换root都无法解决 就是如上所展示的问题,但是程序依然可以启动。 问题解决…...

如何在看板中体现优先级变化
在看板中有效体现优先级变化的关键措施包括:采用颜色或标签标识优先级、设置任务排序规则、使用独立的优先级列或泳道、结合自动化规则同步优先级变化、建立定期的优先级审查流程。其中,设置任务排序规则尤其重要,因为它让看板视觉上直观地体…...

零基础设计模式——行为型模式 - 责任链模式
第四部分:行为型模式 - 责任链模式 (Chain of Responsibility Pattern) 欢迎来到行为型模式的学习!行为型模式关注对象之间的职责分配、算法封装和对象间的交互。我们将学习的第一个行为型模式是责任链模式。 核心思想:使多个对象都有机会处…...
汇编常见指令
汇编常见指令 一、数据传送指令 指令功能示例说明MOV数据传送MOV EAX, 10将立即数 10 送入 EAXMOV [EBX], EAX将 EAX 值存入 EBX 指向的内存LEA加载有效地址LEA EAX, [EBX4]将 EBX4 的地址存入 EAX(不访问内存)XCHG交换数据XCHG EAX, EBX交换 EAX 和 EB…...

Spring数据访问模块设计
前面我们已经完成了IoC和web模块的设计,聪明的码友立马就知道了,该到数据访问模块了,要不就这俩玩个6啊,查库势在必行,至此,它来了。 一、核心设计理念 1、痛点在哪 应用离不开数据(数据库、No…...

均衡后的SNRSINR
本文主要摘自参考文献中的前两篇,相关文献中经常会出现MIMO检测后的SINR不过一直没有找到相关数学推到过程,其中文献[1]中给出了相关原理在此仅做记录。 1. 系统模型 复信道模型 n t n_t nt 根发送天线, n r n_r nr 根接收天线的 MIMO 系…...

Docker 本地安装 mysql 数据库
Docker: Accelerated Container Application Development 下载对应操作系统版本的 docker ;并安装。 基础操作不再赘述。 打开 macOS 终端,开始 docker 安装mysql之旅 第一步 docker search mysql 》〉docker search mysql NAME DE…...

免费PDF转图片工具
免费PDF转图片工具 一款简单易用的PDF转图片工具,可以将PDF文件快速转换为高质量PNG图片。无需安装复杂的软件,也不需要在线上传文件,保护您的隐私。 工具截图 主要特点 🚀 快速转换:本地转换,无需等待上…...
深度学习之模型压缩三驾马车:模型剪枝、模型量化、知识蒸馏
一、引言 在深度学习中,我们训练出的神经网络往往非常庞大(比如像 ResNet、YOLOv8、Vision Transformer),虽然精度很高,但“太重”了,运行起来很慢,占用内存大,不适合部署到手机、摄…...

rknn toolkit2搭建和推理
安装Miniconda Miniconda - Anaconda Miniconda 选择一个 新的 版本 ,不用和RKNN的python版本保持一致 使用 ./xxx.sh进行安装 下面配置一下载源 # 清华大学源(最常用) conda config --add channels https://mirrors.tuna.tsinghua.edu.cn…...