golang:var、new、make区别及使用

new 分配;make 初始化
new(T) 返回 *T 指向一个零值 T
make(T) 返回初始化后的 T,只能用于 slice,map,channel。

先整明白go里面的几大变量“**类型**”(不严谨,只是个人在使用的时候常用到的结构的一个划分)
(1) 值类型 基本类型 +  struct 
(2) 引用类型:主要是map, slice,chan 这三个引用(make创建内存的)
(3)指针类型:那就多了, *int64, *struct…

var 声明

值类型:分配内存空间,并赋该类型的零值

引用类型:不会分配内存,默认就是nil

例子:

//值类型
var i int
fmt.Println(i) //i=0



//引用类型
var j *int  
fmt.Println(j)  
*j = 10  //invalid memory address or nil pointer dereference

//也就是说,空指针还没有内存分配,是不能使用的。

new 分配 

func new(Type) *Type
new(T) 返回 *T 指向一个零值 T   返回指针类型(一般不用)

值类型和引用类型:分配内存空间并赋该类型的零值,返回一个指向该类型内存地址的指针

实际在工程使用中,通常是直接声明指针使用,不需要new操作

例子:

var j *int  
j = new(int)  // 让j里面的内容指向一块分配好的内存地址,地址里面设置int的零值:0
fmt.Println(j)   
fmt.Println(*j)  
*j = 10  
fmt.Println(*j)

make 初始化

func make(t Type, size ...IntegerType) Type
make(T) 返回初始化后的 T   返回引用类型

make用于map, slice,chan 的内存创建,因为他们三个是引用类型,直接返回这三个类型本身(引用类型),不赋零值

例子:

var c chan int //声明管道类型变量c,此时c还是nil,不可用;
fmt.Printf("%#v \n",c) //(chan int)(nil)
c = make(chan int)
fmt.Printf("%#v", c) //(chan int)(0xc000062060)

声明管道类型变量c,此时c还是nil,不可用;
通过make来分配内存并初始化,c就获得了内存可以使用了。

 make(T, args) 返回的值通过函数传递参数之后可以直接修改,即 map,slice,channel 通过函数穿参之后在函数内部修改将影响函数外部的值

func modifySlice(s []int) {
    s[0] = 1
}

s2 := make([]int, 3)
fmt.Printf("%#v", s2) //[]int{0, 0, 0}
modifySlice(s2)
fmt.Printf("%#v", s2) //[]int{1, 0, 0}

 make(T, args) 返回的是引用类型,在函数内部可以直接更改原始值,对 map 和 channel 也是如此。

func modifyMap(m map[int]string) {
    m[0] = "string"
}

func modifyChan(c chan string) {
    c <- "string"
}

m2 := make(map[int]string)
if m2 == nil {
    fmt.Printf("m2 is nil --> %#v \n ", m2) 
} else {
    fmt.Printf("m2 is not nill --> %#v \n ", m2) //map[int]string{}
}

modifyMap(m2)
fmt.Printf("m2 is not nill --> %#v \n ", m2) // map[int]string{0:"string"}


c2 := make(chan string)
if c2 == nil {
    fmt.Printf("c2 is nil --> %#v \n ", c2)
} else {
    fmt.Printf("c2 is not nill --> %#v \n ", c2)
}

go modifyChan(c2)
fmt.Printf("c2 is not nill --> %#v ", <-c2) //"string"

很少需要使用 new

以下代码演示了 struct 初始化的过程,可以说明不使用 new 一样可以完成 struct 的初始化工作。

type Foo struct {
    name string
    age  int
}

//声明初始化
var foo1 Foo
fmt.Printf("foo1 --> %#v\n ", foo1) //main.Foo{age:0, name:""}
foo1.age = 1
fmt.Println(foo1.age)


//struct literal 初始化
foo2 := Foo{}
fmt.Printf("foo2 --> %#v\n ", foo2) //main.Foo{age:0, name:""}
foo2.age = 2
fmt.Println(foo2.age)


//指针初始化
foo3 := &Foo{}
fmt.Printf("foo3 --> %#v\n ", foo3) //&main.Foo{age:0, name:""}
foo3.age = 3
fmt.Println(foo3.age)


//new 初始化
foo4 := new(Foo)
fmt.Printf("foo4 --> %#v\n ", foo4) //&main.Foo{age:0, name:""}
foo4.age = 4
fmt.Println(foo4.age)


//声明指针并用 new 初始化
var foo5 *Foo = new(Foo)
fmt.Printf("foo5 --> %#v\n ", foo5) //&main.Foo{age:0, name:""}
foo5.age = 5
fmt.Println(foo5.age)

foo1 和 foo2 是同样的类型,都是 Foo 类型的值,

foo1 是通过 var 声明,Foo 的 filed 自动初始化为每个类型的零值,

foo2 是通过字面量的完成初始化。

foo3,foo4 和 foo5 是一样的类型,都是 Foo 的指针 Foo。*但是所有 foo 都可以直接使用 Foo 的 filed,读取或修改,为什么?

如果 x 是可寻址的,&x 的 filed 集合包含 m,x.m 和 (&x).m 是等同的go 自动做转换,也就是 foo1.age 和 foo3.age 调用是等价的,go 在下面自动做了转换。

因而可以直接使用 struct literal 的方式创建对象,能达到和 new 创建是一样的情况而不需要使用 new。

  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Golang中的newmake都是用来创建变量的内存空间的,但是它们的使用场景和功能略有不同。 new用于创建值类型(如int、float、struct等)的指针,它返回一个指向新分配的零值的指针。例如,当你需要一个指向某个类型的指针时,可以使用new。具体用法如下: ``` p := new(int) ``` 上面的代码将创建一个int类型的指针p,并将它初始化为0。 make用于创建引用类型(如map、slice、channel等)的对象,它返回一个已初始化的(非零)对象。例如,当你需要创建一个切片时,可以使用make。具体用法如下: ``` s := make([]int, 0, 10) ``` 上面的代码将创建一个长度为0、容量为10的int类型切片s。 因此,newmake的主要区别在于它们创建的变量类型不同,new创建指向零值的指针,而make创建已初始化的引用类型对象。 ### 回答2: 在Go语言中,newmake是两个关键字,用于动态分配空间。它们的作用和使用场景有所不同。 newmake都用于分配内存空间,但是分配的方式和返回值不同。new用于分配值的“零值”空间,返回的是该类型的指针。make用于分配引用类型(如slice、map、channel)的空间,返回的是该类型本身。 具体来说,new关键字用于创建一个指向某个类型的指针,并且分配该类型的“零值”空间。例如,当我们使用new来创建一个指向int类型的指针时,会返回一个指向int类型的零值(即0)的指针。示例代码如下: ``` var i *int i = new(int) fmt.Println(*i) // 输出 0 ``` 而make关键字用于创建引用类型的实例,并且进行初始化。引用类型包括slice、map和channel。使用make创建的实例会进行内部的初始化,并返回一个可以直接使用的实例。示例代码如下: ``` var s []int s = make([]int, 5) fmt.Println(s) // 输出 [0 0 0 0 0] ``` 可以看到,make函数在创建slice实例时进行了初始化,将所有元素初始化为0。这是因为slice是动态大小的数组,需要在内部进行初始化以支持其特殊的特性。 因此,总结起来,new用于创建某个类型的指针并分配“零值”空间,make用于创建引用类型(如slice、map、channel)的实例并进行初始化。 ### 回答3: golang中的newmake都是用于创建内存对象的函数,但它们的用途和功能略有不同。 new是一个内置函数,用于分配内存和初始化类型的零值,并返回指向该类型的指针。可以用new创建值类型(如int、float64等)和引用类型(如结构体、指针等)。示例代码如下: ``` var p *int p = new(int) fmt.Println(*p) // 输出为0,即int类型的零值 ``` make是一个用于创建切片、映射和通道的内置函数,它会分配内存、初始化并返回一个指定类型的已初始化的对象。make只能用于创建引用类型。示例代码如下: ``` var slice = make([]int, 5, 10) fmt.Println(len(slice)) // 输出为5,即切片的长度 fmt.Println(cap(slice)) // 输出为10,即切片的容量 ``` 总结起来,new用于值类型和引用类型的内存分配,返回指向该类型的指针;make用于引用类型(切片、映射和通道)的内存分配和初始化,返回已经初始化的对象。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值