登录
首页 >  Golang >  Go教程

Go中New与Struct初始化区别详解

时间:2026-03-29 20:32:33 278浏览 收藏

在 Go 语言中,`new(MyStruct)` 与 `&MyStruct{}` 虽然都返回指向结构体的指针,但二者能力天差地别:`new` 仅分配零值内存、无法初始化字段、不支持选择性赋值,且在现代 Go 中几乎完全被弃用;而 `&MyStruct{}` 作为复合字面量,既能精准控制字段初始化(包括嵌入字段和导出字段)、支持预设容量或非零值,又能与工具链良好协作(如字段补全、静态检查),更重要的是它能直观暴露 nil 引用类型带来的潜在 panic 风险,促使开发者主动初始化 map、slice 等——简言之,除非维护古董代码,否则永远优先用 `&T{}`,彻底告别 `new`。

如何在Golang中理解New与Struct字面量的区别 Go语言指针初始化方式

什么时候该用 new(),什么时候该用 &MyStruct{}

绝大多数时候,你应该用 &MyStruct{},而不是 new(MyStruct)。前者更直观、可控,且能初始化字段;后者只做零值分配,返回指针,但无法设置任何字段值。

new(T) 的作用非常单一:申请一块 T 类型大小的内存,填满零值,返回 *T。它不调用任何构造逻辑,也不支持字段赋值。而 &MyStruct{Field: val} 既是分配,也是初始化,还能选择性赋值。

  • new(strings.Builder) 返回一个字段全为零的 *strings.Builder,但你没法在调用时指定初始容量或内容
  • &strings.Builder{} 效果相同,但写法更常见;若想预设底层切片容量,只能靠后续调用 Grow()new 完全无能为力
  • 自定义结构体如 type User struct { Name string; Age int }new(User) 得到 &User{Name: "", Age: 0},但无法写成 new(User){Name: "Alice"} —— 语法错误

new() 在现代 Go 代码里几乎没存在必要

Go 1.0 以来,new() 就没被鼓励使用。标准库和主流项目中极少出现它,原因很实在:它提供的能力被字面量完全覆盖,还更啰嗦、更难读。

唯一勉强算“例外”的场景是泛型或反射中需要类型擦除后的零值指针(比如某些 unsafe 或 reflect 操作),但这类代码本就极少见,且通常有更安全的替代方案。

  • new(int) 不如直接写 newInt := new(int); *newInt = 42?其实 newInt := 42 + ptr := &newInt 更清晰,或者一步到位 ptr := &[]int{42}[0](虽然奇怪,但说明没必要)
  • make([]int, 0)new([]int) 完全不同:make 返回可用切片,new 返回指向零值切片头的指针(*[]int),几乎无法直接使用
  • VS Code 或 gopls 对 new(T) 几乎不提供字段补全,而 &T{} 能精准提示可设字段

Struct 字面量加取地址符(&T{})才是默认动作

当你需要一个指向结构体的指针,并希望控制初始化细节时,&T{} 是事实标准。它触发的是复合字面量(composite literal)机制,不是构造函数,但足够灵活。

注意字段顺序无关紧要,但未导出字段(小写开头)在包外不可设;零值字段可省略,非零值必须显式写出(除非用 struct{} 空结构体)。

  • &User{Name: "Tom", Age: 25} 合法;&User{"Tom", 25} 非法(无字段名时必须所有字段按声明顺序给出,且类型需严格匹配)
  • &bytes.Buffer{}new(bytes.Buffer) 运行效果一致,但前者可扩展为 &bytes.Buffer{buf: make([]byte, 0, 1024)}(虽然 buf 是未导出字段,实际不能这么写——这正好说明字段可见性限制比 new 更早暴露问题)
  • 如果结构体含嵌入字段,&T{Embedded: embeddedVal} 可以直接初始化,new(T) 则完全做不到

容易被忽略的零值陷阱:new 不等于“空对象”,而是“全零内存块”

new(T) 分配的是整个 T 大小的连续内存,并全部置零。对包含指针、map、slice、channel 的结构体来说,“零值”不等于“可用”,而是“nil”。这点常被误读。

例如 type Conf struct { Data map[string]int; Log *os.File }new(Conf) 返回的指针所指对象中,Datanil map,Lognil 指针。直接 c.Data["k"] = 1 会 panic,必须先 c.Data = make(map[string]int)

  • &Conf{Data: make(map[string]int)} 一步到位,避免运行时 panic
  • new(Conf) 后忘记初始化内部引用类型,是真实线上 bug 常见来源,尤其在配置结构体或缓存容器中
  • 静态分析工具(如 staticcheck)通常不会报 new(T) 的问题,但会对未初始化的 map/slice 使用发出警告 —— 这进一步削弱了 new 的实用价值
事情说清了就结束。真正要记住的只有两点:别用 new 初始化结构体,除非你在读十年前的老代码;用 &T{} 时,留心哪些字段是 nil,哪些需要手动 make 或 new。

文中关于的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《Go中New与Struct初始化区别详解》文章吧,也可关注golang学习网公众号了解相关技术文章。

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