在 Go 语言中,new 和 make 是两个用于创立目标的内建函数。虽然它们都用于分配内存,但在运用时有一些重要的差异。本文将详细介绍 new 和 make 的差异,并经过多个方面的剖析和代码示例,帮助咱们理解它们的运用场景。

1. 简介

在 Go 语言中,new 和 make 是用于创立目标的两个内建函数,它们的运用方法和效果有所不同。正确理解 new 和 make 的差异关于编写高效的 Go 代码十分重要。下面咱们将逐渐介绍 new 和 make 的详细细节。

2. new 函数

2.1 new 函数的效果

new 函数用于创立一个指定类型的指针,并将其初始化为零值。它回来指向新分配的内存地址的指针。

2.2 new 函数的语法

new 函数的语法十分简单,只需在关键字 new 后边跟上类型即可。

func new(Type) *Type

下面是一个运用 new 函数创立结构体目标的示例代码:

package main
​
import "fmt"type Person struct {
  Name string
  Age int
}
​
func main() {
  p := new(Person)
  fmt.Println(p)
}

在上述代码中,咱们运用 new 函数创立了一个指向 Person 类型的指针 p,并输出其值。运行结果为 &{ 0},表明 p 是一个指向 Person 类型零值的指针。

3. make 函数

3.1 make 函数的效果

make 函数用于创立内建类型(如切片、映射和通道)的目标,它回来一个已初始化的(非零值)目标。

3.2 make 函数的语法

make 函数的语法与 new 函数有所不同,它需求指定类型和一些额定的参数。

func make(Type, size IntegerType) Type

下面是一些运用 make 函数创立切片、映射和通道的示例代码:

package main
​
import "fmt"func main() {
  // 创立一个长度为5,容量为10的整型切片
  s := make([]int, 5, 10)
  fmt.Println(s)
  
  // 创立一个键为string,值为int的映射
  m := make(map[string]int)
  m["a"] = 1
  m["b"] = 2
  fmt.Println(m)
  
  // 创立一个字符串通道
  c := make(chan string)
  fmt.Println(c)
}

在上述代码中,咱们运用 make 函数创立了一个长度为 5,容量为 10 的整型切片 s,一个键为 string,值为 int 的映射 m,并创立了一个字符串通道 c。运行结果分别为 [0 0 0 0 0],map[a:1 b:2] 和 0xc0000460c0,表明这些目标都被初始化为非零值。

4. 差异比照

4.1 分配的类型

  • new 函数用于任何类型的分配,并回来一个指向该类型的指针。
  • make 函数只用于分配切片、映射和通道,并回来初始化后的切片、映射或通道目标。

4.2 回来值类型

  • new 函数回来指向分配类型的指针。
  • make 回来分配类型的初始化后的非零值。

4.3 运用场景

  • new 函数首要用于创立值类型的实例。值类型包含根本类型(如整型、浮点型、布尔型等)以及结构体。new 函数回来一个指向新分配内存的指针,可以方便地对该实例进行操作和修改。

    package main
    ​
    import "fmt"type Point struct {
      X, Y int
    }
    ​
    func main() {
      p := new(Point)
      p.X = 10
      p.Y = 20
      fmt.Println(p) // 输出:&{10 20}
    }
    
  • make 函数首要用于创立引用类型的目标。引用类型包含切片、映射和通道。因为引用类型需求在运用之前进行初始化,make 函数回来的是初始化后的非零值目标,而不是指针。

    package main
    ​
    import "fmt"func main() {
      // 创立一个长度为5,容量为10的整型切片
      s := make([]int, 5, 10)
      fmt.Println(s) // 输出:[0 0 0 0 0]
      
      // 创立一个键为string,值为int的映射
      m := make(map[string]int)
      m["a"] = 1
      m["b"] = 2
      fmt.Println(m) // 输出:map[a:1 b:2]
      
      // 创立一个字符串通道
      c := make(chan string)
      fmt.Println(c) // 输出:0xc0000460c0
    }
    

4.4 示例代码比照

经过下面的示例代码比照,咱们可以更清楚地理解 new 和 make 之间的差异:

package main
import "fmt"type Person struct {
  Name string
  Age int
}
​
func main() {
  // 运用new创立Person类型的指针
  p1 := new(Person)
  p1.Name = "Alice"
  p1.Age = 25
  fmt.Println(p1) // 输出:&{Alice 25}
  // 运用make创立切片
  s1 := make([]int, 5, 10)
  s1[0] = 1
  s1[1] = 2
  fmt.Println(s1) // 输出:[1 2 0 0 0]
}

从示例代码可以看出,运用 new 创立的是指针类型,而运用 make 创立的是初始化后的非零值目标。

5. 总结

在本文中,咱们详细介绍了 Golang 中 new 和 make 的差异及运用场景。经过对两者的语法和示例代码进行剖析,咱们得出以下定论:

  • new 用于任何类型的分配,并回来指向该类型的指针,首要用于创立值类型的实例。
  • make 用于分配切片、映射和通道,并回来初始化后的非零值目标,首要用于创立引用类型的目标。

正确理解 new 和 make 的差异关于编写高效、标准的 Go 代码至关重要。在选择运用 new 还是 make 时,要根据详细的需求和目标类型进行判断。期望本文对咱们有所帮助,可以愈加深化地理解和使用 new 和 make 函数。感谢阅读!