make仅适用于slice、map、channel三种引用类型;因其设计目标是为运行时动态分配容量的类型创建并初始化实例,而数组和struct是编译期大小确定的值类型,无需make。
make 只能用于 slice、map、channel 这三种引用类型,其他类型(包括 struct、array、string、指针等)调用 make 会编译报错。
make?为什么不能用于数组或结构体?make 的设计目标是为「运行时动态分配容量」的引用类型创建实例并预设初始状态。它返回的是引用(底层指向堆内存),而非值本身。
slice:需指定元素类型、长度(len)和可选容量(cap),如 make([]int, 5, 10)
map:只接受类型和可选初始桶数量(hint),如 make(map[string]int, 100);不支持传入长度或容量参数channel:指定元素类型和可选缓冲区大小,如 make(chan int, 8)
[5]i
nt)是值类型,编译期确定大小,直接字面量或变量声明即可,make([5]int) 语法非法new(T) 或字面量初始化;make 对其无意义make 和 new 的关键区别在哪?两者都分配内存,但语义和返回值完全不同:
new(T) 返回 *T,即指向零值的指针,适用于任意类型(包括自定义 struct、int、string 等)make(T, args...) 返回类型 T 本身(非指针),且仅限 slice/map/channel,还会做类型特定初始化(如 map 的哈希表准备、slice 的底层数组分配)make([]*int, 5) 创建的是长度为 5 的 slice,每个元素是 *int 类型的零值(即 nil 指针),不是 5 个已分配的 *int
编译器对 make 参数检查严格,错一个就报错,典型现象如下:
make([]int, -1) → panic: makeslice: len out of range(运行时 panic,非编译错误)make(map[int]string, "wrong") → 编译失败:cannot use "wrong" (type string) as type int in argument to make
make(chan int, 3.14) → 编译失败:cannot use 3.14 (type float64) as type int in argument to make
make([]int, 5, 3) → 编译失败:cap is smaller than len(容量不能小于长度)合理设置容量/提示值能减少后续扩容或 rehash 开销,但过度预估反而浪费内存:
make([]byte, 0, expectedSize) 配合 append 更高效make(map[int]int, 1000) 比默认更少触发扩容,但 hint=0 或省略也完全合法package main
import "fmt"
func main() {
s := make([]int, 3, 5) // len=3, cap=5
m := make(map[string]bool, 8) // hint=8
c := make(chan string, 2) // buffer size=2
fmt.Println(len(s), cap(s)) // 3 5
fmt.Println(len(m)) // 0
fmt.Println(cap(c)) // 2
}
最容易被忽略的一点:make 返回的不是地址,不能对其取地址(&make([]int, 5) 是非法语法);如果需要指针,得先赋值再取址。