七叶笔记 » golang编程 » Golang面试make和new的用法

Golang面试make和new的用法

在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分配的空间。

相关文章