new用于分配任意类型的零值内存并返回指针,make仅用于初始化slice、map、channel并返回实例本身,二者用途和返回值不同。

在Go语言中,new 和 make 都是用于内存分配的内置函数,但它们的用途和行为完全不同。理解它们的区别对掌握Go的内存管理机制非常重要。
new 的作用与原理
new(T) 是一个通用的内存分配函数,它为类型 T 分配一块零值的内存空间,并返回指向该类型的指针 *T。
它的主要特点包括:
- 只做内存分配,不初始化复杂结构(如slice、map、channel)
- 分配的空间会被初始化为类型的零值(例如 int 为 0,指针为 nil)
- 适用于任意类型,但返回的是指向零值的指针
例如:
立即学习“go语言免费学习笔记(深入)”;
ptr := new(int) *ptr = 10 // ptr 指向一个初始值为 0 的 int,现在被赋值为 10
对于自定义结构体也很常见:
type Person struct {
Name string
Age int
}
p := new(Person)
// p 指向一个字段都为零值的 Person 实例
make 的作用与原理
make(T, args...) 并不分配对象本身,而是用于初始化 slice、map 和 channel 这三种引用类型,并返回类型 T 本身(不是指针)。
它的核心功能是:
- 仅能用于 slice、map、channel
- 完成内部数据结构的初始化(如底层数组、哈希表、缓冲队列等)
- 返回的是类型实例,而不是指针
例如:
立即学习“go语言免费学习笔记(深入)”;
slice := make([]int, 5) // 创建长度为5的切片 m := make(map[string]int) // 创建空的 map ch := make(chan int, 10) // 创建带缓冲的 channel
如果尝试用 make 初始化其他类型,比如结构体或基本类型,编译器会报错。
关键区别总结
- 目标类型不同:new 可用于任意类型;make 只能用于 slice、map、channel
- 返回值不同:new 返回指针 *T;make 返回类型 T 本身
- 初始化程度不同:new 只分配并清零;make 会构造运行时所需的数据结构
- 使用场景不同:new 常用于需要显式指针的地方;make 是使用引用类型的必要步骤
举个对比例子:
var p *[]int = new([]int) // p 是指向 slice 的指针,slice 本身为空 s := make([]int, 5) // s 是长度为5的 slice,已准备就绪可用
此时 *p 是一个零值 slice(nil),而 s 已经可以正常使用下标赋值。
底层实现简析
从运行时角度看,new 调用的是 Go 的内存分配器(mallocgc),分配指定大小的堆内存并清零。它本质上是一个泛型的内存申请函数。
make 在编译期间会被转换成特定的运行时函数调用:
- make([]T, len) → runtime.makeslice
- make(map[K]V) → runtime.makemap
- make(chan T) → runtime.makechan
这些函数不仅分配内存,还设置类型信息、初始化哈希表、创建锁结构等,确保类型可正常工作。
基本上就这些。new 和 make 看似相似,实则职责分明:new 是通用内存分配器,make 是引用类型的初始化工具。正确使用它们,是写出高效、安全Go代码的基础。










