Golang面试make和new的用法

周游列国之仕子 2020-09-15

在golang中,make和new都分配内存,但是它们之间仍然存在一些差异。只有了解它们之间的差异,才能在适当的场合使用它们。

简而言之,new只是分配内存,而不初始化内存;make分配并初始化内存。所谓的初始化就是给一个类型赋一个初始值,例如,字符为空,整数为0,逻辑值为false。

从Golang的官方文档的builtin(内置的软件包)中可以找到,make和new的用法。

Golang面试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) 

相关推荐