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

new 返回的是指针,且不初始化底层数据
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 只用于 slice、map、channel,返回对应类型值而非指针
make 专为三种内置引用类型设计:[]T、map[K]V、chan T。它分配底层数据结构(如数组、哈希表、环形缓冲区),并返回**已初始化的值本身**,不是指针。
立即学习“go语言免费学习笔记(深入)”;
关键点:
-
make([]int, 3)返回[]int,底层数组已分配,长度=3,容量=3 -
make(map[string]int)返回map[string]int,非 nil,可直接myMap["k"] = 1 -
make(chan int, 10)返回chan int,带缓冲区 - 不能对 struct、int、string 等调用
make,编译失败
容易踩的坑:误用 make 初始化自定义类型(哪怕它内部含 slice/map)——必须用字面量或构造函数。
为什么不能用 new 创建 map 或 slice?
因为 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,何时选 make?
绝大多数情况你不需要 new。Go 推荐显式初始化(字面量、make、构造函数),而非依赖 new 的零值指针。
真实选择逻辑:
- 要 slice/map/channel → 必须用
make - 要非 nil 的 struct 指针且字段全为零值 → 可用
new(T),但更常见写法是&T{} - 要带初始字段的 struct 指针 → 只能用
&T{Field: val}或工厂函数 - 性能上无实质差异,但
make对 slice/map/channel 有额外开销(如哈希表初始化),new就是纯内存分配
最容易被忽略的一点:new 和 make 都不触发 GC 扫描(它们分配在堆上,但由编译器静态决定),但这和你的使用选择无关——关键是别混淆类型契约。










