Go中的make和new的区别

Python019

Go中的make和new的区别,第1张

·new:是一个用来分配内存的内置函数,与C++不同的是,它不初始化内存,只是将其归零,也就相当于,new(X)会为X的新项目分配被归零的存储,且返回它的地址,其中,第一个参数是类型,返回值是类型的指针,其值被初始化为‘0’,对于不同的数据类型,0值的意义也是不一样的,比如int初始化为0,bool初始化为false等等。

·make:是Golang的内置函数,仅用于分配和初始化slice、map及channel类型的对象,三种类型都是结构,返回值为类型而不是指针,例如slice是一个三元描述符,包含一个指向数据(在数组中)的指针,长度以及容量,在这些项被初始化前,slice都是nil的,对于这三者,make初始化这些内部数据结构,并准备好可用的值。

需要注意的是,make只用于map、slice和channel,并且不反悔指针,想要获得一个显式的指针,使用new进行分配,或者显式地使用一个变量的地址。

new 主要用于结构体的初始化

make用于数组array,切片slice,协程chnnel的初始化

例如: users:=make([10]int)

msg:=make(chan int)

new会分配结构空间,并初始化为清空为零,不进一步初始化

new之后需要一个指针来指向这个结构

make会分配结构空间及其附属空间,并完成其间的指针初始化

make返回这个结构空间,不另外分配一个指针

例子new:

var p *[]int = new([]int)

p := new([]int)

以上分配了一个slice结构,但是结构中的应该指向底层数组的ptr指针为空,故实际不能往这个slice里面存取数据

同时分配了一个指针p,也即(在32位系统中)占4个字节并存放slice结构的地址

例子make:

var v []int = make([]int, 0)

v := make([]int, 0)

以上分配了一个slice结构,且结构中的应该指向底层数组的ptr指针已经指向了某个底层数组,这个底层数组应该已经分配了,故这个slice已经可以使用了

注意v就是这个slice结构,而不是一个指向slice的指针

上述仅是示例,一般使用时都会明确长度和容量:v := make([]int, 10, 50)

结论:

由上可见,用new来分配slice的意义不大,因为没有恰当的初始化,无法直接使用

有附带空间的结构,使用make来初始化,可以完成内部指针初始化,其后可以立即使用

它们都是内置函数,new按指定类型长度分配零值内存,返回指针,并不关系类型内部构造和初始化方式。而引用类型则必须使用make函数创建,编译器会将make转化为目标类型专用的创建函数(或指令),以保证完成全部内存分配和相关属性初始化。

new可以为引用类型分配内存,但这是不完整创建。以字典(map)为例,它仅分配了字典类型本身(实际就是个指针包装)所需内存,并没有分配键值存储内存,也没有初始化散列桶等内部属性,因此它无法正常工作。

p := new(map[string]int) // 函数new返回指针

m := *p

m["a"] = 1 // panic: assignment to entry in nil map (运行期错误

使用make初始化引用类型

切片的存储形式 unsafe.pointer len cap

只声明不赋值时

使用make初始化时 会分配底层数组

使用new初始化引用类型

使用new初始化引用类型 (new的返回值就是slice结构的起始地址,不会分配底层数组,所以此时赋值时不允许的)

使用append的方式添加元素,可以分配底层数组

参考: 【Golang】slice类型存什么?make和new?slice和数组?扩容规则?_哔哩哔哩_bilibili