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 时,将其加入结果集…...
QModMaster:5分钟掌握免费开源ModBus调试工具终极指南
QModMaster:5分钟掌握免费开源ModBus调试工具终极指南 【免费下载链接】qModbusMaster 项目地址: https://gitcode.com/gh_mirrors/qm/qModbusMaster 你是否在为工业设备调试而烦恼?面对复杂的ModBus通信协议,商业软件价格昂贵&#…...
如何用Bypass Paywalls Clean突破付费墙限制?技术解析与实战指南
如何用Bypass Paywalls Clean突破付费墙限制?技术解析与实战指南 【免费下载链接】bypass-paywalls-chrome-clean 项目地址: https://gitcode.com/GitHub_Trending/by/bypass-paywalls-chrome-clean 在数字内容付费墙日益严密的今天,Bypass Payw…...
续航提升40%?EnergyStarX让Windows 11设备电量焦虑成为历史
续航提升40%?EnergyStarX让Windows 11设备电量焦虑成为历史 【免费下载链接】EnergyStarX 🔋 Improve your Windows 11 devices battery life. A WinUI 3 GUI for https://github.com/imbushuo/EnergyStar. 项目地址: https://gitcode.com/gh_mirrors/…...
从零构建32位MIPS单周期处理器:Logisim实战与24条核心指令实现详解
1. 从零理解MIPS单周期处理器 第一次接触CPU设计时,我盯着教科书上的数据通路图看了整整三天——那些密密麻麻的连线和缩写让我头晕目眩。直到用Logisim动手搭建了一个最简单的加法器,才突然明白处理器不过是精心设计的电子积木。单周期MIPS处理器就像乐…...
投资分析太复杂?用TradingAgents-CN实现零代码智能分析的3个方案
投资分析太复杂?用TradingAgents-CN实现零代码智能分析的3个方案 【免费下载链接】TradingAgents-CN 基于多智能体LLM的中文金融交易框架 - TradingAgents中文增强版 项目地址: https://gitcode.com/GitHub_Trending/tr/TradingAgents-CN TradingAgents-CN作…...
Qwen3-14B GPU算力优化实践:显存占用降低28%的FlashAttention-2配置
Qwen3-14B GPU算力优化实践:显存占用降低28%的FlashAttention-2配置 1. 开箱即用的私有部署方案 对于想要快速部署Qwen3-14B大模型的企业和个人开发者来说,这个经过优化的私有部署镜像提供了完美的解决方案。它基于RTX 4090D 24GB显存显卡和CUDA 12.4环…...
Redis 单线程真的是单线程吗?源码角度全面解析
Redis 是单线程的——这句话流传太广了,以至于很多人真的以为 Redis 就一个线程在跑。但实际上,如果你 ps -ef 或者 top 看一眼正在运行的 Redis 进程,会发现线程数不止一个。 到底怎么回事?这篇文章从源码角度把这个问题彻底说清…...
PS插件加载失败?手把手教你用注册表修复PS2017-2022扩展未签署问题
PS插件加载失败?手把手教你用注册表修复PS2017-2022扩展未签署问题 当你在Photoshop中安装新插件时,突然弹出"扩展未经正确签署"的错误提示,这种挫败感我深有体会。作为一名长期与PS插件打交道的设计师,这个问题几乎成…...
全能解析工具UniExtract2:多格式提取的效率革命
全能解析工具UniExtract2:多格式提取的效率革命 【免费下载链接】UniExtract2 Universal Extractor 2 is a tool to extract files from any type of archive or installer. 项目地址: https://gitcode.com/gh_mirrors/un/UniExtract2 在数字化信息处理领域&…...
高性能无线基带FPGA实现:开源802.11 WiFi实时信号处理架构解析
高性能无线基带FPGA实现:开源802.11 WiFi实时信号处理架构解析 【免费下载链接】openwifi open-source IEEE 802.11 WiFi baseband FPGA (chip) design: driver, software 项目地址: https://gitcode.com/gh_mirrors/op/openwifi Openwifi是一个基于软件定义…...
