今天在聊make和new函数之前,咱们先来看一种现象 当然如果你对golang的指针还不是很了解,可以先看看这篇文章,以做到无缝连接

。看完这个,我彻底理解了golang的指针。

case1​

var a int
a = 1
fmt.Println(a)
//返回结果
//1

这个例子很简单,咱们定义了一个int类型的变量,然后赋值直接输出,这个没什么好疑问的,但是咱们把这个例子改一下

case2​

  var p *int
*p = 10
fmt.Println(*p)

请问上面代码会输出什么?不出意外会输出10对不对,但是咱们运行之后,命令行却输出

panic: runtime error: invalid memory address or nil pointer dereference
[signal SIGSEGV: segmentation violation code=0x2 addr=0x0 pc=0x10428df44]

goroutine 1 [running]:
main.main()

这是为什么呢?

简单说说原因:

图片

因为我们在var p *int,只是把p变量定义为了指针类型,p的默认值为nil,「并没有在内存上分配对应的空间」,既然没有空间,那给p变量赋值自然就会报错,当然p任然是占空间的,只不过p的值为nil

如果我们把代码改成下面这样:

 var p *int = new(int)
*p = 10
fmt.Println(*p)
//输出结果
//10

结果就可以正常输出,为什么加个new就可以正常输出呢?想要回答这个问题,首先你得知道new函数做了什么。

图片

new函数主要做了下面三件事

  • 调用操作系统接口申请一块int类型的内存空间
  • 操作系统将分配的空间返回给go程序
  • 在内存中开辟了一块int的空间,并且把该空间的指针指向p 因为现在指针p已经指向了一块被分配的空间,所以可以直接进行赋值操作

小结​

对于int,string,float,rune,byte,bool等类型,在定义变量的时候系统已经给申请了内存,而且给了对应的默认值(int的默认值为0,string的默认值为"",bool的默认值为false),所以我们可以直接给变量进行赋值操作。

对于指针,切片,map等类型,这些变量直接定义的时候系统是没有给分配内存的,并且默认值为nil,所以不能直接赋值。如果想赋值的话,需要用new或者make函数向系统提前申请内存才行。

make和new的区别​

make和new都是用来内存分配的方法,简单的说,new只分配内存,

  • 「make用于slice,map,和channel的初始化,不仅可以开辟一个内存,还能给这个内存的类型初始化其零值」。
  • make返回的还是引用类型本身;而new返回的是指向类型的指针。

make

func make(t Type, size ...IntegerType) Type 

new

func new(Type) *Type

make只能用来分配及初始化类型为slice,map,channel的数据;new可以分配任意类型的数据。