• make使用场景 slice map channel
  • new 使用场景 大部分使用在结构体当中

1. new 和 make

执行报错

执行下面的代码会引发 panic,为什么呢?

  • 在 Go 语言中对于引用类型的变量,我们在使用的时候不仅要声明它,还要为它分配内存空间,否则我们的值就没办法存储。
  • 而对于值类型的声明不需要分配内存空间,是因为它们在声明的时候已经默认分配好了内存空间。
  • 要分配内存,就引出来今天的 new 和 make。Go 语言中 new 和 make 是内建的两个函数,主要用来分配内存。
func main() {
var userinfo map[string]string
    userinfo["username"] = "张三"
    fmt.Println(userinfo)
}
/*
panic: assignment to entry in nil map
*/

2. make和new比较

new 和 make 是两个内置函数,主要用来创建并分配类型的内存。

make和new区别

  • make 关键字的作用是创建 slice、map 和 channel 等内置的数据结构
  • new 的作用是为类型申请一片内存空间,并返回指向这片内存的指针(申请空间和返回指针)
func main() {
a := make([]int, 1, 10) // 切片长度为 1,预留空间长度为 10
a = append(a,1)
fmt.Printf("%v--%T \n",a,a) // [0 0 0]--[]int 值----切片本身
var b = new([]int)
//b = b.append(b,2) // 返回的是内存指针,所以不能直接 append
*b = append(*b, 3) // 必须通过 * 指针取值,才能进行 append 添加
fmt.Printf("%v--%T",b,b) // &[]--*[]string 内存的指针---内存指针
}

3. new函数

系统默认的数据类型,分配空间

// 1.new实例化int
age := new(int)
*age = 1
// 2.new实例化切片
li := new([]int)
*li = append(*li, 1)
// 3.实例化map
userinfo := new(map[string]string)
*userinfo = map[string]string{}
(*userinfo)["username"] = "张三"
fmt.Println(userinfo) // &map[username:张三]
}

自定义类型使用 new 函数来分配空间

func main() {
var s *Student
s = new(Student) //分配空间
s.name ="zhangsan"
fmt.Println(s) // &{zhangsan 0}
}
type Student struct {
name string
age int
}

4. make函数

  • make 也是用于内存分配的,但是和 new 不同,它只用于 chan、map 以及 slice 的内存创建而且它返回的类型就是这三个类型本身,而不是他们的指针类型。
  • 因为这三种类型就是引用类型,所以就没有必要返回他们的指针了
package main
import "fmt"
func main() {
a := make([]int,1,10) // 切片长度为 1,预留空间长度为 10
b := make(map[string]string)
c := make(chan int, 1)
fmt.Println(a,b,c) // [0 0 0] map[] 0xc0000180e0
}

当我们为slice分配内存的时候,应当尽量预估到slice可能的最大长度,通过给make传第三个参数的方式来给slice预留好内存空间,这样可以避免二次分配内存带来的开销,大大提高程序的性能。

您可能感兴趣的文章: