反射修改变量必须传入地址,因reflect.Value默认只读;需用reflect.ValueOf(&x).Elem()获取可寻址值,且字段须导出、类型匹配,通过CanSet()校验而非CanAddr()。
Go 语言的 reflect.Value 默认是只读副本,直接对 reflect.ValueOf(x) 返回的值调用 Set* 方法会 panic,错误信息类似:reflect.Value.SetString using unaddressable value。这是因为

MapIndex)、数组/结构体字段(当整个结构体可寻址时)才满足条件。
实操建议:
reflect.ValueOf(&x).Elem(),再调用 SetInt、SetString 等Field(i) 返回的仍是不可修改的副本*T 类型参数不是所有“看起来可寻址”的值都能被反射修改。常见失败点:
reflect.ValueOf("hello").Addr() 会 panic —— 字符串字面量不可取地址reflect.ValueOf([]int{1,2,3}).Index(0) 返回的 Value 不可修改,因为底层数组是临时分配且不可寻址;需先取切片地址:reflect.ValueOf(&s).Elem().Index(0)
map 中通过 MapIndex 获取的 Value 默认不可修改(即使 map 是指针),必须用 SetMapIndex 替代直接赋值CanSet() 返回 false,这是 Go 的导出规则限制,与可寻址性无关CanSet(),别信 CanAddr()
CanAddr() 表示该 Value 是否有地址(比如是否来自指针解引用),但它不保证能修改;CanSet() 才是最终判断依据——它内部已综合检查了可寻址性 + 字段导出性 + 是否为不可变类型(如 unsafe.Pointer)。
安全写法示例:
func setString(v reflect.Value, s string) error {
if !v.CanSet() {
return fmt.Errorf("cannot set value: %v", v)
}
if v.Kind() == reflect.String {
v.SetString(s)
return nil
}
return fmt.Errorf("not a string")
}
注意:即使 v.CanAddr() 为 true,若它是未导出字段或来自常量,CanSet() 仍为 false。
反射修改变量本质是绕过编译期检查,运行时开销大,且极易因类型不匹配或权限不足 panic。生产环境应严格限制使用场景:
最易被忽略的一点:reflect.Value 的 Set* 方法不会自动做类型转换,SetInt(42) 对 int64 字段有效,对 int32 就 panic —— 必须先用 Convert() 或确保 Kind 和类型完全匹配。