失眠网,内容丰富有趣,生活中的好帮手!
失眠网 > golang未初始化的slice和初始化为空的slice的区别简析

golang未初始化的slice和初始化为空的slice的区别简析

时间:2020-04-06 20:56:20

相关推荐

golang未初始化的slice和初始化为空的slice的区别简析

先看两种形式的slice定义

// 定义未初始化的slicevar s1 []string// 通过字面量形式定义并初始化为空slicevar s2 = []string{}// 通过make函数定义并初始化为空slicevar s3 = make([]string, 0, 0)

我们知道,在golang中切片是对底层数组中的连续一部分存储空间的引用,是类似一种数组指针的存在,例如我们可以直接像打印一个指针所指向的地址的形式来打印切片

// 先来看%p 打印格式的用法var arr = [...]string{"aaa"}fmt.Printf("%p\n", arr) // 输出 %!p([1]string=[aaa]),指示这并不是一个指针fmt.Printf("%p\n", &arr) // 正常输出地址 0xc0001021e0fmt.Printf("s1: %p, s2: %p, s3: %p\n", s1, s2, s3) //输出s1: 0x0, s2: 0x119f408, s3: 0x119f408 说明在golang中slice的底层就是指针,s1的值其实就是nilif s1 == nil {fmt.Println("s1 yes") // 输出 s1 yes}if s2 == nil {fmt.Println("s2 yes") // 没输出}if s3 == nil {fmt.Println("s3 yes") // 没输出}

从上面可以看出,s1没有具体指向的内存空间, 而s2和s3都指向了具体的内存空间,这就是它们的最大区别,而s2和s3没有区别,只是两种不同的书写形式罢了。其实以上三种形式在golang代码中的使用基本上是没有什么区别的,例如

fmt.Println(s1, s2, s3) //[] [] []fmt.Println(len(s1), cap(s1), len(s2), cap(s2), len(s3), cap(s3)) //0 0 0 0 0 0for i, v := range s1 {fmt.Println(i, v)} // 没报错,没有输出for i, v := range s2 {fmt.Println(i, v)} // 没报错,没有输出for i, v := range s3 {fmt.Println(i, v)} // 没报错,没有输出s1 = append(s1, "s1s1")s2 = append(s2, "s2s2")s3 = append(s3, "s3s3")fmt.Println(s1, s2, s3) // [s1s1] [s2s2] [s3s3]s1 = s1[:] // 不报错s2 = s2[:] // 不报错s3 = s3[:] // 不报错func Print(s... string) {if len(s) > 0 {fmt.Println(s[0])} else {fmt.Println("[]")}}Print(s1...) // 不报错Print(s2...) // 不报错Print(s3...) // 不报错

所以,在正常的使用过程中,如果无法预知slice的大小需要一个空的slice,使用s1的形式是没有问题的,而且能减少一次不必要的内存分配

如果觉得《golang未初始化的slice和初始化为空的slice的区别简析》对你有帮助,请点赞、收藏,并留下你的观点哦!

本内容不代表本网观点和政治立场,如有侵犯你的权益请联系我们处理。
网友评论
网友评论仅供其表达个人看法,并不表明网站立场。