new返回未初始化的指针,仅适用于值类型;make返回已初始化的slice/map/channel值,专用于这三种引用类型。

new 的作用很简单:分配一块内存,类型对齐,返回指向该内存的指针,但**不做任何初始化(即值为零值)**。它只接受一个类型参数,不能用于 slice、map、channel 这类引用类型(编译报错)。
常见错误是以为 new([]int) 能创建可直接使用的切片——实际会报 cannot use new([]int) (value of type *[]int) as []int value in assignment,因为返回的是 *[]int,不是 []int。
适用场景极少,比如需要一个非 nil 的结构体指针但又不想调用构造函数时:
type User struct {
Name string
Age int
}
u := new(User) // u 是 *User,u.Name == "",u.Age == 0
make 专为三种内置引用类型设计:[]T、map[K]V、chan T。它分配底层数据结构(如数组、哈希表、环形缓冲区),并返回**已初始化的值本身**,不是指针。
关键点:
make([]int, 3) 返回 []int,底层数组已分配,长度=3,容量=3make(map[string]int) 返回 map[string]int,非 nil,可直接 myMap["k"] = 1
make(chan int, 10) 返回 chan int,带缓冲区make,编译失败容易踩的坑:误用 make 初始化自定义类型(哪怕它内部含 slice/map)——必须用字面量或构造函数。
因为 map 和 slice 在 Go 中是**描述符(descriptor)**,本质是结构体,包含指针、长度、容量等字段。它们的零值是 nil,但 nil map/slice 无法直接使用(如 append 或 map[key] = val 会 panic)。
new(map[string]int) 返回 *map[string]int,即一个指向 nil map 的指针;解引用后仍是 nil,没解决根本问题。而 make(map[string]int) 直接返回一个已分配哈希表的 map[string]int 值。
对比示例:
var m1 map[string]int = new(map[string]int) // ❌ 错误:m1 是 *map[string]int,*m1 是 nil m2 := make(map[string]int // ✅ 正确:m2 是可用的 map
绝大多数情况你不需要 new。Go 推荐显式初始化(字面量、make、构造函数),而非依赖 new 的零值指针。
真实选择逻辑:
make
new(T),但更常见写法是 &T{}
&T{Field: val} 或工厂函数make 对 slice/map/channel 有额外开销(如哈希表初始化),new 就是纯内存分配最容易被忽略的一点:new 和 make 都不触发 GC 扫描(它们分配在堆上,但由编译器静态决定),但这和你的使用选择无关——关键是别混淆类型契约。