在golang中,make和new都分配内存,但是它们之间仍然存在一些差异。只有了解它们之间的差异,才能在适当的场合使用它们。
简而言之,new只是分配内存,而不初始化内存;make分配并初始化内存。所谓的初始化就是给一个类型赋一个初始值,例如,字符为空,整数为0,逻辑值为false。
从Golang的官方文档的 builtin( 内置的软件包)中可以找到,make和new的用法。
new的使用介绍:
我们先来看看new的定义
// 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.
//内建函数new分配内存。其第一个实参为类型,而非值。其返回值为指向该类型的新分配的零值的指针。
func new(Type) *Type
可以看出,它的参数是一个类型,返回值是指向该类型的内存地址的指针,并且分配的内存将被设置为零,即该类型的零值,即字符为空,整数为0,逻辑值为false
看一些例子
type P struct {
Name string
Age int
}
var a *[2]int
var s *string
var b *bool
var i *int
var ps *P
a = new([2]int)
s = new(string)
b = new(bool)
i = new(int)
ps = new(P) //structure
fmt.Println(a, " ", *a)
fmt.Println(s, " ", *s)
fmt.Println(b, " ", *b)
fmt.Println(i, " ", *i)
fmt.Println(ps, " ", *ps)
输出如下:
&[0 0] [0 0]
0xc0000821e0
0xc0000a409a false
0xc0000a40b0 0
&{ 0} { 0}
上面基础类型,我们看一下slice, map and channel类型是如何操作的:
//map 操作
var mp *map[string]string
mp = new(map[string]string)
//注释掉下面的行,new map 返回为nil,直接使用会panic
//*mp = make(map[string]string) // if this line is omitted, it will pan "Pan: assignment to entry in nil map"“
(*mp)["name"] = "lc"
fmt.Println((*mp)["name"])
// slice 操作
var ms *[]string
ms = new([]string)
// 注释掉下面的行访问的时候会下标超出范围
//*ms = make([]string,5) // if this line is deleted, it will "panic: runtime error: index out of range"
(*ms)[0] = "lc"
fmt.Println((*ms)[0])
从上面可以看出,silce,map,channel和其他类型是引用类型。当引用类型初始化为nil时,不能直接分配nil,也不能使用new来分配内存,还需要使用make来进行分配。
make的使用介绍:
我们看一下make的定义
/ /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. The specification of the result depends on
// the type:
// Slice: The size specifies the length. The capacity of the slice is
// equal to its length. A second integer argument may be provided to
// specify a different capacity; it must be no smaller than the
// length. For example, make([]int, 0, 10) allocates an underlying array
// of size 10 and returns a slice of length 0 and capacity 10 that is
// backed by this underlying array.
// Map: An empty map is allocated with enough space to hold the
// specified number of elements. The size may be omitted, in which case
// a small starting size is allocated.
// Channel: The channel's buffer is initialized with the specified
// buffer capacity. If zero, or the size is omitted, the channel is
// unbuffered.
//切片:size指定了其长度。该切片的容量等于其长度。切片支持第二个整数实参可用来指定不同的容量; 它必须不小于其长度,因此 make([]int, 0, 10) 会分配一个长度为0,容量为10的切片。
//映射:初始分配的创建取决于size,但产生的映射长度为0。size可以省略,这种情况下就会分配一个小的起始大小。
//通道:通道的缓存根据指定的缓存容量初始化。若 size为零或被省略,该信道即为无缓存的。
func make(t Type, size ...IntegerType) Type
可以看出,它返回的是类型本身而不是指针类型,因为make只能为slice,map,channel等初始化内存,并且它们返回引用类型,因此不必返回指针
让我们看一些make的例子:
mm :=make(map[string]string)
mm["name"] = "lc"
fmt.Println(mm["name"])
mss :=make([]int,2)
mss[0] = 100
fmt.Println(mss[0])
ch :=make(chan int,1)
ch <-100
fmt.Println(<-ch)
总结:
make仅用于分配和初始化slice,map和chan类型的数据。new可以分配任何类型的数据
。new分配返回一个指针,即Type * Type。make返回一个引用,该引用为Type
由make分配的空间之后,清除并初始化由new分配的空间。