登录
首页 >  Golang >  Go教程

Golang指针与内存分配:new与make详解

时间:2026-01-04 22:53:44 137浏览 收藏

大家好,我们又见面了啊~本文《Golang指针与内存分配详解:new与make用法解析》的内容中将会涉及到等等。如果你正在学习Golang相关知识,欢迎关注我,以后会给大家带来更多Golang相关文章,希望我们能一起进步!下面就开始本文的正式内容~

new()只分配零值内存并返回指针,make()专用于slice、map、channel的初始化;用new(map[string]int会得到nil指针导致panic,必须用make()创建可使用的实例。

如何使用Golang理解指针与内存分配_Golangnew和make使用方法

new() 和 make() 的根本区别在哪

new() 只做一件事:在堆上分配零值内存,返回指向该内存的指针;make() 专用于内置引用类型(slicemapchannel),不仅分配内存,还完成初始化(比如 slicelencap 设置、map 的底层哈希表构建)。

常见错误是试图用 new(map[string]int)new([]int) —— 编译会通过,但得到的是一个 nil 值,后续操作直接 panic:

var m = new(map[string]int
m["key"] = 1 // panic: assignment to entry in nil map

这是因为 new() 返回的是 *map[string]int,而你拿到的是一个指向 nil 的指针,不是可使用的 map 实例。

什么时候必须用 make() 而不能用 new()

只要你要创建一个可立即使用的 slicemapchannel,就必须用 make()

  • make([]int, 5) → 长度为 5、容量为 5 的切片,元素全为 0
  • make(map[string]bool, 10) → 预分配约 10 个桶的 map(第二个参数是 hint,非精确容量)
  • make(chan int, 100) → 缓冲区大小为 100 的 channel

注意:make() 不接受指针类型作为第一个参数,make(*[]int, 5) 是非法语法;它只认原始类型 []Tmap[K]Vchan T

new() 的典型使用场景和陷阱

new() 的价值在于快速获得一个「已分配、已清零、可取地址」的变量指针,适合结构体或自定义类型的初始化起点:

type Config struct {
    Port int
    Host string
}
p := new(Config) // 等价于 &Config{},p 是 *Config,字段全为零值

但它不适用于需要「非零初始状态」的场景。比如你想创建一个长度为 3 的切片并预设值,new([]int) 给你 *[]int(即 nil 指针),完全无法下标赋值;必须用 make([]int, 3) 或字面量 []int{1,2,3}

另一个易错点:new(int) 返回 *int,其值是 0 的地址;但 new(int8)new([4]byte) 同理 —— 它们不关心类型是否是复合类型,只管按类型大小分配并清零。

如何验证内存分配行为(调试小技巧)

fmt.Printf("%p", &x) 查看变量地址,配合 unsafe.Sizeofreflect.ValueOf(x).IsNil() 判断是否为 nil,能快速区分 newmake 的结果:

var s1 = new([]int)
var s2 = make([]int, 0)
fmt.Printf("s1 addr: %p, isNil: %t\n", s1, reflect.ValueOf(*s1).IsNil()) // true
fmt.Printf("s2 addr: %p, isNil: %t\n", &s2, reflect.ValueOf(s2).IsNil()) // false

真正容易被忽略的是:Go 的栈逃逸分析可能让看似在栈上的 new 分配实际落在堆上——你无法仅从 newmake 字面判断最终内存位置,得靠 go build -gcflags="-m" 看编译器决策。

本篇关于《Golang指针与内存分配:new与make详解》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于Golang的相关知识,请关注golang学习网公众号!

前往漫画官网入口并下载 ➜
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>