Go反射指南

简介: 反射与接口息息相关

 概念:

官方对此有个非常简明的介绍,两句话耐人寻味:

  1. 反射提供一种让程序检查自身结构的能力
  2. 反射是困惑的源泉

第1条,再精确点的描述是“反射是一种检查interface变量的底层类型和值的机制”。 第2条,很有喜感的自嘲,不过往后看就笑不出来了,因为你很可能产生困惑。

reflect 实现了运行时的反射能力,能够让程序操作不同类型的对象。反射包中有两对非常重要的函数和类型,两个函数分别是:

  • reflect.TypeOf()   能获取类型信息;
  • reflect.ValueOf() 能获取数据的运行时表示;

只有这么简单吗?当然不是,请继续阅读。

引出:

其实了解反射的第一步,应从interface入手,因为反射与接口存在着千丝万缕的关系。

如下是一段interface的源码

type iface struct {
    tab  *itab
    data unsafe.Pointer
}
// layout of Itab known to compilers
// allocated in non-garbage-collected memory
// Needs to be in sync with
// ../cmd/compile/internal/gc/reflect.go:/^func.dumptypestructs.
type itab struct {
    inter  *interfacetype
    _type  *_type
    link   *itab
    bad    int32
    inhash int32      // has this itab been added to hash?
    fun    [1]uintptr // variable sized
}

image.gif

看不懂也没关系,我对其大致简化一番,从reflect角度再来看看,并思考从iface中看到的字段:

type I interface{
    // 方法集
}
type iface struct{
    typ reflect.Type   // 储存类型信息
    val reflect.Value  // 储存实际值
}

image.gif

之所以引出interface,是因为想说interface类型有个(value,type)对,而反射就是检查interface的这个(value, type)对的。具体一点说就是Go提供一组方法提取interface的value,提供另一组方法提取interface的type。

  • reflect.Type 提供一组接口处理interface的类型,即(value, type)中的type
  • reflect.Value 提供一组接口处理interface的值,即(value, type)中的value

下面会提到反射对象,所谓反射对象即反射包里提供的两种类型的对象。

  • reflect.Type 类型对象
  • reflect.Value 类型对象

三大法则:

第一法则:

interface{} 变量,可以反射出反射对象;

下面示例,看看是如何通过反射获取一个变量的值和类型的:

package main
import (
    "fmt"
    "reflect"
)
func main() {
    var x float64 = 3.4
    t := reflect.TypeOf(x)  //t is reflext.Type
    fmt.Println("type:", t)
    v := reflect.ValueOf(x) //v is reflext.Value
    fmt.Println("value:", v)
}
运行如下:
type: float64
value: 3.4

image.gif

是不是疑惑了,明明是上述是x->reflect类型,却依然说是 interface{} --变为--> reflect类型呢?这是因为,在TypeOf 与 ValueOf 内部,自动将 值类型,转化为了 接口类型。

第二法则:

从反射对象可以获取 interface{} 变量;

package main
import (
    "fmt"
    "reflect"
)
func main() {
    var x float64 = 3.4
    v := reflect.ValueOf(x) //v is reflext.Value
    var y float64 = v.Interface().(float64)
    fmt.Println("value:", y)
}

image.gif

1、用reflect.ValueOf(x) 获取,value值。

2、v.Interface() 转化成接口。

3、类型断言转化成,对应的基本类型

第三法则:

要修改反射对象,其值必须可设置。

通过反射可以将interface类型变量转换成反射对象,可以使用该反射对象设置其持有的值。在介绍何谓反射对象可修改前,先看一下失败的例子:

package main
import (
    "reflect"
)
func main() {
    var x float64 = 3.4
    v := reflect.ValueOf(x)
    v.SetFloat(7.1) // Error: will panic.
}
如下代码,通过反射对象v设置新值,会出现panic。报错如下:
panic: reflect: reflect.Value.SetFloat using unaddressable value

image.gif

错误原因即是v是不可修改的。

反射对象失败,取决于是否可以修改其储存的值。回想一下函数传参时,是传值还是传址,就不难理解上例中为何失败。

上例中,传入 reflect.ValueOf() 函数的其实是x的值,而非x本身。即通过v修改其值是无法影响x的,也即是无效的修改,所以 golang 会报错。

想到此处,即可明白,如果构建v时使用x的地址就可实现修改了,但此时v代表的是指针地址,我们要设置的是指针所指向的内容,也即我们想要修改的是*v。 那怎么通过v修改x的值呢?

reflect.Value 提供了 Elem() 方法,可以获得指针向指向的Value 。看如下代码:

package main
import (
"reflect"
    "fmt"
)
func main() {
    var x float64 = 3.4
    v := reflect.ValueOf(&x)
    v.Elem().SetFloat(7.1)
    fmt.Println("x :", v.Elem().Interface())
}

image.gif

1、调用reflect.ValueOf 获取变量指针。

2、调用 reflect.Value.Elem 获取指针指向的变量。

3、调用 reflect.Value.SetFloat() 更新变量。

总结:

以上为本篇博客精华内容,如有不妥,请及时私信联系我,斟酌之后必加以纠正。

待后续深入学习时,会转回继续修改。

参考内容:

1、《Go专家编程》

2、《Go语言设计与实践》

3、sql生成器-实战


目录
相关文章
|
24天前
|
人工智能 Linux API
阿里云/本地部署OpenClaw多Bot群内协作指南:一键配置提示词+大模型API完整方案及避坑指南
OpenClaw真正强大的地方,在于支持多Bot在同一群组内自动协作,通过1个Boss Bot+多个执行Bot的分工模式,实现一句话下发需求、全流程自动执行、自动汇总结果。本文将复杂的多智能体配置完全整理为可直接使用的配置与提示词,同时提供2026年阿里云部署、MacOS/Linux/Windows11本地部署流程,以及阿里云千问大模型API、免费Coding Plan API配置方法,搭配全套代码命令与常见问题解答,让你无需理解复杂配置,即可拥有一支全自动AI协作团队。
622 4
|
25天前
|
缓存 安全 API
从零精通OpenClaw保姆级教程:阿里云/本地部署、api配置、Skill实战与性能优化全攻略
OpenClaw作为开源AI助手平台的核心代表,凭借高度灵活性与可扩展性,已成为个人高效办公、团队协作自动化的关键工具。其通过模块化的Skill生态,可实现文件管理、网络搜索、任务自动化、代码开发等多元化场景需求,无需复杂开发即可搭建专属智能工作流。
1061 2
|
11天前
|
Ubuntu 算法 关系型数据库
Debian/Ubuntu 环境 PolarDB-X 单机版 DEB 包安装综合指南
本文整合阿里云文档,详解Ubuntu 18.04与Debian 10下PolarDB-X单机版安装:因官方仅提供RPM包,需用alien转DEB,但二者压缩格式不同(Ubuntu用zstd,Debian 10不支持),必须在目标系统本地转换,不可复用。含依赖处理、配置初始化及启动验证全流程。
284 19
|
1月前
|
存储 人工智能 关系型数据库
OpenClaw怎么可能没痛点?用RDS插件来释放OpenClaw全部潜力
OpenClaw插件是深度介入Agent生命周期的扩展机制,提供24个钩子,支持自动注入知识、持久化记忆等被动式干预。相比Skill/Tool,插件可主动在关键节点(如对话开始/结束)执行逻辑,适用于RAG增强、云化记忆等高级场景。
885 56
OpenClaw怎么可能没痛点?用RDS插件来释放OpenClaw全部潜力
|
29天前
|
关系型数据库 MySQL 数据安全/隐私保护
MySQL 8.0安装教程 Windows版:解压+自定义组件+传统认证+密码设置+命令行快捷方式创建指南
MySQL是开源、高性能、高可靠的关系型数据库。本文详解MySQL 8.0.17的下载、解压、自定义安装及传统认证方式配置,并指导创建命令行快捷方式,快速完成本地部署与验证。(239字)
|
27天前
|
人工智能 自然语言处理 监控
【养龙虾保姆级教程】OpenClaw是什么?能做什么?怎么部署?
“养龙虾”是开发者对开源AI智能体框架OpenClaw的昵称——它能在本地运行,理解自然语言并直接操控电脑执行任务(如办公、开发、爬虫等),堪称可自托管的“数字员工”。本文带你零基础掌握其原理、能力与安全部署方法。
642 10
|
27天前
|
机器学习/深度学习 人工智能 监控
2000张人脸眼部检测数据集分享(适用于目标检测任务已标注+划分)
本数据集含2000张人脸图像,已精细标注人脸及左右眼位置,按1400/400/200划分为训练/验证/测试集;支持YOLO/VOC/COCO格式,覆盖多场景、光照、姿态与人群属性,标注误差<2像素,开箱即用,适用于疲劳检测、注意力分析等任务。
191 4
|
24天前
|
人工智能 JavaScript Linux
OpenClaw从零到一部署手册:本地+阿里云环境搭建+大模型API完整配置教程
OpenClaw是一款轻量化、可扩展、本地优先的AI智能体框架,能够通过自定义人格、记忆、工具与技能,打造完全属于个人的AI助手。本文提供最简洁、稳定、可复现的安装流程,覆盖Windows11、MacOS、Linux三大本地系统,并补充2026年阿里云服务器部署方案,同时包含阿里云千问大模型API与免费Coding Plan API的完整配置方法,搭配所有必备代码命令与常见问题解决方案,让任何人都能在5分钟内完成环境搭建,启动属于自己的AI助手。
1377 4
|
26天前
|
缓存 安全 测试技术
GO项目开发规范文档解读
本篇博客的目的,更多是为快速翻阅与回忆使用。
174 1