Go中结构体字段导出性仅由首字母大小写决定:大写可导出,小写不可导出;嵌入字段提升、反射访问、JSON序列化均受此规则约束。
Go 语言中结构体字段能否被其他包访问,只看字段名首字母是否大写。小写字母开头的字段是私有的,即使结构体本身可导出,该字段也无法从外部包直接读写。
type User struct { Name string; age int }:外部包能读写 Name,但无法访问 age
public/private 关键字,全靠命名约定控制可见性inner),其字段即使大写也无法导出当结构体嵌入另一个类型(如 time.Time 或自定义结构体)时,Go 会尝试“提升”其导出字段到外层结构体作用域,但仅限于字段名不冲突、且被嵌入类型是导出的。
time.Time 后,可直接写 u.Year(),因为 Year 是导出方法,且无同名字段/方法冲突ID 字段,必须显式通过 u.EmbedA.ID 或 u.EmbedB.ID 访问,否则编译报错 ambiguous selector u.ID
inner)时,其字段和方法不会被提升,只能
通过 u.inner.field 访问(且仍受限于导出规则)标准库 reflect 包默认禁止读写未导出字段。强行操作会导致 panic:reflect: reflect.Value.Interface: cannot return value obtained from unexported field or method。
reflect.Value.CanAddr() && reflect.Value.CanInterface() 判断是否可安全取值reflect.Value.Addr().Interface() 获取指针,再调用 .SetXxx();否则 panicu := User{Name: "Alice", age: 25}
v := reflect.ValueOf(&u).Elem()
ageField := v.FieldByName("age")
if ageField.CanAddr() && ageField.CanInterface() {
ageField.SetInt(26)
}
JSON 标准库不依赖字段导出性,而依赖结构体标签(json:"...")。未导出字段加上 json:"field_name" 标签后,json.Unmarshal 可以成功赋值,但 json.Marshal 仍不会输出该字段。
type Config struct { Port int `json:"port"`; token string `json:"token"` }:反序列化时 token 能被设值,但序列化结果里不会有 "token" 字段-(如 json:"-"),则该字段既不参与序列化也不参与反序列化omitempty(如 json:"name,omitempty")只影响序列化,不影响反序列化逻辑