登录
首页 >  Golang >  Go教程

Go语言struct结构体定义教程

时间:2026-03-31 12:54:26 124浏览 收藏

本文深入解析Go语言结构体的核心使用规范与常见陷阱:强调字段首字母大小写决定导出性(大写才可跨包访问、参与JSON序列化和ORM映射),揭示小写字段在跨包调用、API响应和数据库操作中“静默失效”的典型问题;详解嵌入结构体需以导出类型为基础才能实现字段提升,避免因匿名字段未导出导致访问失败;力推键值对初始化方式以杜绝字段错位风险;厘清方法接收者选择逻辑——值类型适用于只读场景,指针类型保障修改生效且更高效,并建议统一风格;最后点明嵌入本质是组合而非继承,澄清同名字段冲突与方法歧义的处理机制。这些看似细微的规则,恰恰是写出健壮、可维护Go代码的关键所在。

Go语言结构体如何定义_Go语言struct结构体教程【实战】

结构体字段名必须大写才能被外部访问

Go 没有 public/private 关键字,字段是否可导出(即能被其他包访问)只看首字母:大写(NameID)是导出字段,小写(nameid)是包内私有。这点不注意,跨包调用时会直接报 cannot refer to unexported field

  • JSON 序列化时,小写字段自动被忽略(json.Marshal 不会包含它们)
  • ORM(如 GORM)映射数据库时,同样跳过小写字段,容易导致数据漏存
  • 嵌入结构体时,如果嵌入的是小写结构体(如 person),它不会提升字段,s.Name 会编译失败

初始化结构体优先用字段名赋值,别依赖顺序

用值列表初始化(Student{101, "lisi", 20, 97})看似简洁,但一旦结构体字段增删或顺序调整,所有这类初始化都会 silently 错位——比如把 Age 写进 Score 位置,编译不报错,运行时才出逻辑 bug。

  • 推荐始终使用键值对初始化:Student{ID: 101, Name: "lisi", Age: 20, Score: 97}
  • 字段可以乱序,也可以省略(省略字段取零值:""0nil
  • 加了新字段后,旧初始化代码仍能编译通过,不会意外覆盖其他字段

方法接收者选值类型还是指针类型?看要不要改原值

定义方法时,接收者写 func (s Student) Modify() 还是 func (s *Student) Modify(),不是风格问题,而是语义和行为差异。

  • 值接收者:方法内对 s 的任何修改(如 s.Age = 25)都不会影响调用方的原始结构体实例
  • 指针接收者:能真正修改原值,也更省内存(避免拷贝整个结构体)
  • 只要结构体中任一方法用了指针接收者,建议全部统一用指针——否则调用方混用 s.Method()&s.Method() 容易混乱
  • 结构体含 slice/map/chan 等引用字段时,即使只读,也建议用指针接收者,避免误以为值拷贝是深拷贝

嵌入结构体不是继承,是字段提升 + 方法组合

type Employee struct { Person; Title string },看起来像“继承”,但 Go 里没有继承概念。它只是把 Person 的导出字段和方法“提上来”,让 e.Namee.SayHello() 可以直接调用。

  • 如果嵌入的是未命名字段(如 Person),就叫“匿名字段”,支持字段提升;如果写了名字(如 person Person),就得用 e.person.Name 访问
  • 两个嵌入结构体有同名导出字段(比如都含 ID),则 e.ID 会编译报错,必须显式写成 e.Person.IDe.Company.ID
  • 方法冲突不报错,但调用时若两个嵌入类型都有 Save()e.Save() 会编译失败,必须限定调用路径

最容易被忽略的是字段可见性与序列化的联动:一个字段明明在代码里看着能读写,但 json.Marshal 输出为空,十有八九是首字母小写了;还有嵌入结构体时忘了它本身也得是导出类型(Person 而非 person),否则提升失效。

本篇关于《Go语言struct结构体定义教程》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于Golang的相关知识,请关注golang学习网公众号!

资料下载
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>