

新闻资讯
技术教程IsExported() 是判断字段是否导出的唯一标准方式,返回 true 表示首字母大写、可被其他包通过反射读写,false 则不可见且无法安全访问。
IsExported() 判断字段是否导出最直接Go 中字段是否“可导出”,本质就是看它首字母是否大写——这是编译器级的可见性规则,反射不能绕过,但可以检测。调用 reflect.StructField.IsExported() 是唯一标准、安全、推荐的方式。
true:字段名首字母大写(如 Name、ID),其他包可通过反射读/写(前提是结构体本身也导出且值可寻址)false:字段名小写(如 age、email),即使你拿到 reflect.Value,CanInterface() 和 CanSet() 也几乎必为 false,强行读可能 panic,写则直接失败IsExported() 只作用于 reflect.StructField,不是 reflect.Value;得先用 reflect.TypeOf(x).FieldByName("xxx") 或遍历 NumField() 拿到字段描述,再调用FieldByName() 找不到小写字母字段?这不是反射“没找到”,而是 Go 的设计约束:未导出字段在反射层面就不可见。调用 reflect.Value.FieldByName("age") 会返回零值(reflect.Value{}),且 IsValid() 为 false,不是报错,容易被忽略。
v := reflect.ValueOf(user); f := v.FieldByName("age"); if !f.IsValid() { ... } —— 这里 f 必然无效,不是代码写错了,是语言规则如此unsafe 或绕过反射去访问:虽技术上可能,但破坏封装、不可移植、不兼容 future Go 版本,生产环境严禁Age()),而不是改字段名CanSet() 不等于“能修改”,它依赖导出性 + 可寻址性双重校验CanSet() 返回 true 仅当两个条件同时满足:字段已导出(IsExported() == true),且你持有的是它的可寻址反射值(比如传入的是 &user 而非 user)。
reflect.ValueOf(user).FieldByName("Name").CanSet() → 总是 false,因为 ValueOf(user) 是副本,不可寻址u := &user
v := reflect.ValueOf(u).Elem() // 必须 Elem() 得到结构体本身
f := v.FieldByName("Name")
if f.IsValid() && f.CanSet() {
f.SetInt(100)
}User.Profile.Age)要逐层检查:每级字段都必须导出,且每级 Value 都需可寻址,否则 CanSet() 立即失败
回事,但 json 包只处理导出字段字段有没有 json:"name" 标签,跟它是否导出完全无关;但像 json.Marshal() 这类标准库函数,内部用反射时只遍历导出字段——所以未导出字段哪怕打了 tag,也会被静默忽略。
type User { name string `json:"name"` } 序列化结果是 {},不是 {"name":"x"}
json 包调用的是 reflect.Value.Field(i),而 i 只覆盖导出字段索引范围(NumField() 返回值不包含未导出字段)t := reflect.TypeOf(User{})
for i := 0; i < t.NumField(); i++ {
f := t.Field(i)
fmt.Printf("%s: exported=%t, tag=%q\n", f.Name, f.IsExported(), f.Tag)
} 你会发现小写字段根本不会出现在循环里