Go反射是运行时动态获取类型、值、调用方法等的能力,依赖interface{}和运行时类型信息;reflect.TypeOf返回类型描述,reflect.ValueOf返回可操作的值,设值需可寻址,结构体字段修改须用.Elem(),tag需Tag.Get()提取,方法调用需导出且接收者匹配。

reflect 包提供。它不是语法糖,也不是编译期特性,而是一套严格依赖接口值(interface{})和运行时类型信息的机制——用错一步,就 panic;少判一个条件,就静默失败。
reflect.TypeOf 和 reflect.ValueOf 怎么用?区别在哪?
这两个函数是反射的入口,但行为完全不同:reflect.TypeOf 返回 reflect.Type,只描述“是什么类型”;reflect.ValueOf 返回 reflect.Value,代表“这个值本身”,且附带可操作性。
-
reflect.TypeOf(42)→int类型描述,不能取值或设值 -
reflect.ValueOf(42)→ 一个reflect.Value,可用.Int()、.String()等方法读值,但不能直接设值 - 要设值,必须传指针:
reflect.ValueOf(&x).Elem(),否则.CanSet()永远返回false - 如果传的是 nil 接口值(比如
var x interface{}),reflect.ValueOf(x)的.IsValid()为false,后续调用会 panic
为什么修改结构体字段总失败?CanSet() 到底在检查什么?
反射设值失败最常见的原因是忽略了“可寻址性”——reflect.Value 必须来自一个可寻址的变量(通常是取地址后的指针),否则 .CanSet() 返回 false,调用 .SetXxx() 会 panic。
- 错误写法:
v := reflect.ValueOf(u); v.FieldByName("Name").SetString("A")→ panic:reflect: cannot set unaddressable value - 正确写法:
v := reflect.ValueOf(&u).Elem(); v.FieldByName("Name").SetString("A") - 即使结构体字段是导出的(首字母大写),如果原始值不可寻址,依然无法设值
- 对 map、slice、chan 等引用类型,
.SetXxx()仅适用于其元素(如slice[i]),而非容器本身
结构体 tag 怎么安全读取?常见误用有哪些?
字段 tag 是字符串字面量(如 `json:"name"`),必须通过 reflect.StructField.Tag.Get("key") 提取,不是直接访问字段属性。
- tag 值必须是反引号包裹的字符串,双引号或单引号会编译失败
-
Tag.Get("json")返回空字符串不等于不存在——可能只是没写该 key,需结合Tag.Lookup("json")判断是否存在 - 若结构体是指针,
reflect.TypeOf(u)得到的是*User类型,需先.Elem()才能遍历字段 - tag 解析无自动校验,拼错 key(如
Tag.Get("jsom"))只会返回空,容易埋下静默 bug
反射调用方法为什么总报 “invalid method” 或 “cannot call unexported method”?
反射只能调用导出方法(首字母大写),且接收者类型必须匹配:值接收者方法可被值或指针反射对象调用;指针接收者方法必须用指针反射对象(reflect.ValueOf(&obj))。
- 错误:
reflect.ValueOf(obj).MethodByName("Save")→ 若Save是(*T).Save,会 panic - 正确:
reflect.ValueOf(&obj).MethodByName("Save") - 方法参数必须全部是
reflect.Value,且类型、数量、顺序严格匹配,否则.Call()panic -
.MethodByName()返回的reflect.Value若.IsValid() == false,说明方法不存在,应提前检查










