Go初始化struct有四种方式:①字面量需显式赋值所有字段(含未导出字段);②new(T)和&T{}生成零值指针,后者支持字段名初始化;③NewXXX构造函数封装校验与默认值;④嵌入结构体须用字段名避免歧义。
Go 中初始化 struct 最直观的方式是用字面量,但字段顺序和可导出性直接影响能否编译通过。如果结构体含未导出字段(小写开头),且在包外初始化,struct{} 字面量会直接报错 cannot refer to unexported field。
即使在包内,漏掉某个字段(尤其非指针类型)也会导致编译失败——Go 不允许跳过非零值字段:
type User struct {
Name string
Age int
addr string // 小写,包内可用,但不能在字面量中省略
}
u := User{"Alice", 25} // ❌ 编译错误:too few values in struct literal
正确写法必须显式写出所有字段(或用字段名+值):
u := User{Name: "Alice", Age: 25, addr: "hidden"}(仅限包内)u := User{Name: "Alice", Age: 25} ✅ 若 addr 是指针或可设为零值字段(如 *string 或 addr *string)new(T) 返回指向零值 T 的指针,适用于只想获得一个空结构体指针、后续再逐个赋值的场景。它不调用任何构造逻辑,也不支持带参数

&T{} 效果类似,但更常用,且支持字段名初始化(避免顺序依赖):
u := new(User) → *User,所有字段为零值(Name=="", Age==0, addr=="")u := &User{Name: "Bob"} → 字段名初始化,未写的字段自动为零值,安全且清晰u := &User{"Bob", 30} → ❌ 危险!依赖字段顺序,一旦结构体加字段就易崩注意:new() 返回的是指针,&T{} 也是指针,二者语义一致,但后者更符合 Go 社区习惯。
当结构体需要默认值、字段校验、资源预分配或隐藏内部字段时,应提供 NewXXX 函数。这不是语言特性,而是 Go 的惯用模式。
例如强制 Name 非空、Age 合理:
func NewUser(name string, age int) (*User, error) {
if name == "" {
return nil, errors.New("name cannot be empty")
}
if age < 0 || age > 150 {
return nil, errors.New("invalid age")
}
return &User{
Name: name,
Age: age,
addr: "default", // 封装内部默认值
}, nil
}
关键点:
*User 而非 User:避免大结构体拷贝,也方便后续方法接收指针接收者当 struct 嵌入另一个 struct 时,字面量初始化会按“嵌入字段展开后”的顺序处理。若两个嵌入字段有同名字段,初始化时会冲突或静默覆盖。
例如:
type Person struct {
Name string
}
type Employee struct {
Person
Name string // ❌ 同名字段,编译报错:duplicate field Name
修正后(避免同名):
type Employee struct {
Person
Title string
}
e := Employee{Person: Person{Name: "Alice"}, Title: "Dev"}
但更常见且安全的做法是只用字段名初始化:
e := Employee{Person: Person{Name: "Alice"}, Title: "Dev"} ✅ 明确e := Employee{Name: "Alice", Title: "Dev"} ❌ 错误:Go 会尝试赋给 Employee.Name,但该字段不存在(除非显式声明)嵌入带来的初始化歧义,往往在重构时才暴露——建议优先用字段名初始化,避免依赖隐式展开。