xdm 我是阿兵云原生
使用 golang 进行编码过程中,你们都是如何初始化数据结构的呢
GO 里面提供了 2 个关键字来进行初始化数据结构
- new
- make
不过有时候我也会直接使用花括号的方式直接就初始化,例如这样:
代码语言:javascript复制name := []string{"xiaomotong","bob","marry"}
上面这种情况,是初始化的时候已经明确好了需要填的值,可以使用上面这种做法,不过工作中大多数情况初始化的时候,数据还是未知的,大部分时间还是会使用上述两种方式
make
make 是 golang 提供的关键字,主要是用于初始化内置的数据结构,不能初始化自定义的数据结构 , make 可以初始化的数据类型有:
- slice 切片
- map hash表
- channel 通道
例如我们初始化的时候就可以这样写
- 初始化一个切片,填入 len 值, cap 值,切片底层的数据结构除了这俩值以外,还有一个指针,指向的是一个底层数组
- 初始化一个 map,map 的底层是指向一个 hmap 的结构体指针,结构体里面的具体成员感兴趣的 xdm 可以翻看我的历史文章
- 初始化一个通道 ch,这个 ch 是指向一个 hchan 的结构体指针,同样也可翻看我的历史文章看细节
strs := make([]string, 0, 10)
myMap := make(map[string]string, 10)
ch := make(chan struct{}, 10)
我们查看 make 的源码解释可以看到,make 确实支持创建和初始上述 3 种类型
根据解释,我们知道,make 的返回值的类型与填入参数的类型保持一致,咱们填入的是切片类型,那么返回值就是切片类型,填入其他类型对应返回值也是我们期望的类型,没毛病
new
那么为什么有了 make 做数据结构的初始化,还需要 new 干啥?
细心的 胖鱼 可以看出来,前面说到了 make 只适用于初始化 golang 提供的内置数据结构,对于自定义的数据结构,就捉襟见肘了 , 这个时候还是需要 new 出马摆平了
用 new 初始化数据结构可以这样写:
- 自定义一个结构体,里面 2 个成员, age 和 name
- 使用 new 对 T 最初始化,返回一个结构体的指针
type T struct{
age int
name string
}
func main(){
t := new(T)
fmt.Println("t.name == ",t.name)
fmt.Println("t.age == ",t.age)
}
程序执行效果如下:
代码语言:javascript复制# go run main.go
t.name ==
t.age == 0
果然是给我初始化了数据结构,只不过数据都是 零值
当然我们也可以这样写:
代码语言:javascript复制func main(){
a := new(int)
fmt.Println("a == ",a)
fmt.Println("*a == ",*a)
}
程序执行效果如下:
代码语言:javascript复制# go run main.go
a == 0xc420018078
*a == 0
根据程序运行结果,我们知道 new 返回的是一个指针,且这个指针指向的内存地址上对应的值是一个该类型的 零值
瞅一眼 new 源码说明
image-20211017150756595
golang 源码中说到, new 的第一个参数是 数据类型,不是一个值,并且 new 的返回值是一个根据参数类型分配零值内存的指针
因此,我们上面看到的 new 出来的数据结构,打印出来的数据都是对应类型的零值
make 和 new 的区别
最后再来梳理一下 make 和 new 的区别,加深印象
- make 只能用来初始化内置数据类型
- slice
- map
- chan 而 new 可以分配任意类型的数据(包括自定义数据类型)
- make 返回的是引用,是 Type 类型,new 返回的是指针,是 *Type 类型的
日拱一卒,每天写一点点
欢迎点赞,关注,收藏
朋友们,你的支持和鼓励,是我坚持分享,提高质量的动力
好了,本次就到这里
技术是开放的,我们的心态,更应是开放的。拥抱变化,向阳而生,努力向前行。
我是阿兵云原生,欢迎点赞关注收藏,下次见~