golang 中有两个内置函数可以用来在堆上分配内存,分别是 mak() 和 new(),这篇文章中会简要介绍下这两个函数的区别,以及这两个函数分别在什么情况下使用。

下面先看下 golang 中对这两个内置函数的定义。

new():

func new(Type) *Type

The new built-in function allocates memory. The first argument is a type, not a value, and the value returned is a pointer to a newly allocated zero value of that type.

make();

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

The make built-in function allocates and initializes an object of type slice, map, or chan (only). Like new, the first argument is a type, not a value. Unlike new, make's return type is the same as the type of its argument, not a pointer to it.

从上面两个函数的定义,我们可以看到两个明显的区别。

区别一:接收的参数个数不一样

从 new() 函数的定义可以看到,它只接收一个参数 —— Type,即要为哪种类型分配内存。

而从 make() 函数的定义可以看到,它可以接收多个参数,第一个参数也是 Type,剩下的是可选的整数类型参数。

make() 函数可以为 slice 类型分配内存,当 make() 函数为 slice 类型分配内存时,可选的整型参数可以用来指定 slice 的 length 和 capacity,示例如下:

package main

import (
    "fmt"
)

func main() {
    my_slice := make([]int, 5, 10)
    fmt.Println(my_slice)
}

运行上面的程序,会得到如下输出结果:

[0 0 0 0 0]

可以看到,golang 为我们分配了一个长度为 5 的 slice。

区别二:返回类型不一样

new() 函数返回一个指向接收参数类型的指针。

make() 函数返回类型和它接收的第一个参数类型一样。

下面我们通过一个例子来验证这一点:

package main

import (
    "fmt"
)

func main() {
    new_int := new(int)
    fmt.Printf("new_int type: %T\n", new_int)

    make_slice := make([]int, 5)
    fmt.Printf("make_slice: %T\n", make_slice)
}

运行上面的程序,输出结果如下:

new_int type: *int
make_slice: []int

可以看到,new_int 的类型是一个指向int类型的指针;而 make_slice 是一个包含int类型的 slice。

区别三:应用场景不一样

make() 函数专门用来为 slice、map、chan 来分配内存并做初始化的;而 new() 更像是一个通用的内存分配工具,可以为其他类型分配内存。

上面说到 new() 更像是一个通用的内存分配工具,但是它不可以为 slice、map、chan 这种引用类型来分配内存。这是因为 new() 只是简单的分配内存,并在内存上根据类型做零值(zero value)处理。

像slice、map、chan这样的引用类型,在创建这样类型的值后,会创建一个称作 header 的值,它其实是一个数据结构,这个数据结构中,包含一个指向底层数据结构的的指针,另外根据引用类型的不同,还会包含不同的字段用来管理底层数据结构。

以 slice 类型为例,当创建一个 slice 类型的值时,会创建一个轻量级的数据结构,这个数据结构包含三个字段:pointer、length、capacity。

  • pointer 是一个指向底层数组(underly array)的指针。
  • length 代表 slice 的长度
  • capacity 代表 slice 的容量

由此可以看到为什么不能用 new() 来创建 slice、map、chan 这样的引用类型了。如果用 new() 来创建 slice,那么创建的 header 中的 pointer 做0值处理,就会被初始化为 nil,而 length 和 capacity 也会被初始化为0,这样显然是不正确的。

总结

new() 和 make() 都是 golang 中用来分配内存的函数,本文介绍了 make() 和 new() 的三个主要的区别:

  • 接收参数个数不一样:new() 只接收一个参数,而 make() 可以接收多个参数
  • 返回类型不一样:new() 返回一个指针,而 make() 返回类型和它接收的第一个参数类型一样
  • 应用场景不一样:make() 专门用来为 slice、map、chan 这样的引用类型分配内存并作初始化,而 new() 用来为其他类型分配内存。

一如既往,如果你对文章中的内容有任何疑问,或者是发现文章中有任何错误,都可以通过留言告诉我;如果你喜欢我的文章,欢迎关注我的微信公众号 Tech For Geek。