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都无法解决 就是如上所展示的问题,但是程序依然可以启动。 问题解决…...
Zustand 状态管理库:极简而强大的解决方案
Zustand 是一个轻量级、快速和可扩展的状态管理库,特别适合 React 应用。它以简洁的 API 和高效的性能解决了 Redux 等状态管理方案中的繁琐问题。 核心优势对比 基本使用指南 1. 创建 Store // store.js import create from zustandconst useStore create((set)…...
关于nvm与node.js
1 安装nvm 安装过程中手动修改 nvm的安装路径, 以及修改 通过nvm安装node后正在使用的node的存放目录【这句话可能难以理解,但接着往下看你就了然了】 2 修改nvm中settings.txt文件配置 nvm安装成功后,通常在该文件中会出现以下配置&…...
关于 WASM:1. WASM 基础原理
一、WASM 简介 1.1 WebAssembly 是什么? WebAssembly(WASM) 是一种能在现代浏览器中高效运行的二进制指令格式,它不是传统的编程语言,而是一种 低级字节码格式,可由高级语言(如 C、C、Rust&am…...
佰力博科技与您探讨热释电测量的几种方法
热释电的测量主要涉及热释电系数的测定,这是表征热释电材料性能的重要参数。热释电系数的测量方法主要包括静态法、动态法和积分电荷法。其中,积分电荷法最为常用,其原理是通过测量在电容器上积累的热释电电荷,从而确定热释电系数…...
算法岗面试经验分享-大模型篇
文章目录 A 基础语言模型A.1 TransformerA.2 Bert B 大语言模型结构B.1 GPTB.2 LLamaB.3 ChatGLMB.4 Qwen C 大语言模型微调C.1 Fine-tuningC.2 Adapter-tuningC.3 Prefix-tuningC.4 P-tuningC.5 LoRA A 基础语言模型 A.1 Transformer (1)资源 论文&a…...
学习一下用鸿蒙DevEco Studio HarmonyOS5实现百度地图
在鸿蒙(HarmonyOS5)中集成百度地图,可以通过以下步骤和技术方案实现。结合鸿蒙的分布式能力和百度地图的API,可以构建跨设备的定位、导航和地图展示功能。 1. 鸿蒙环境准备 开发工具:下载安装 De…...
Tauri2学习笔记
教程地址:https://www.bilibili.com/video/BV1Ca411N7mF?spm_id_from333.788.player.switch&vd_source707ec8983cc32e6e065d5496a7f79ee6 官方指引:https://tauri.app/zh-cn/start/ 目前Tauri2的教程视频不多,我按照Tauri1的教程来学习&…...
__VUE_PROD_HYDRATION_MISMATCH_DETAILS__ is not explicitly defined.
这个警告表明您在使用Vue的esm-bundler构建版本时,未明确定义编译时特性标志。以下是详细解释和解决方案: 问题原因: 该标志是Vue 3.4引入的编译时特性标志,用于控制生产环境下SSR水合不匹配错误的详细报告1使用esm-bundler…...
Linux中INADDR_ANY详解
在Linux网络编程中,INADDR_ANY 是一个特殊的IPv4地址常量(定义在 <netinet/in.h> 头文件中),用于表示绑定到所有可用网络接口的地址。它是服务器程序中的常见用法,允许套接字监听所有本地IP地址上的连接请求。 关…...
