打开APP
userphoto
未登录

开通VIP,畅享免费电子书等14项超值服

开通VIP
一篇文章带你了解Go语言基础之切片补充
锦城虽云乐,不如早还家。 上篇文章中我们学习了Go语言基础中的变量,一篇文章带你了解Go语言基础之变量,这篇文章我们继续介绍Go语言基础知识,今天跟大家分享的是基础数据类型之切片补充,一起来学习下吧~
前言
Hey,大家好呀,我是星期八,这次咱们继续学习Go基础之切片补充扒。
make疑云
我们知道,可以通过make创建切片。
var names = make([]string,10,10)这句话表示动态创建了一个切片,切片中的元素数量为10个,切片的容量也为10个。
你有疑惑吗???
切片的数量和容量是什么???
他俩什么关系???
切片本质
其实切片,终究是一个存储数据的一个东西,目前知道数组是可以存储东西的。
其实切片的本质,还是数组,只不过是Go帮助我们做了一些封装,可以方便的对切片里面的数据增删改查。
例如:
package main
import "fmt"
func main() { var names = make([]int, 4, 10) //int类型默认值是0 fmt.Println(names, len(names), cap(names)) //结果:[0 0 0 0] 4 10}理解图。
没错,本质就是指向了一个长一点的数组。
但是这个数组是会自动扩容的,当容量(cap)append满了之后,会自动扩容。
现在,我们就知道make里面参数的意义了。
注意:在Go中,推荐使用make创建切片,并且在创建时,需要考虑容量,尽可能不触发容量自动扩容机制,提高性能。
为什么切片append之后,前面会有空格
在上一章中,大概有这样一段代码。
package main
import "fmt"
func main() { var names = make([]int,5,10) names = append(names,11,23,231) fmt.Println(names)//[0 0 0 0 0 11 23 231]}append之后,前面会有很多0,这是怎么回事。
解释:
在通过make创建切片时,第二个参数是切片元素的数量。
上述代码切片第二个参数是5,表示在创建切片时,前5个就已经有值了,只不过是int默认值0。
所以再append时,是再原有的基础上,添加值的,直到cap满了之后,触发扩容机制。
如图所示。
现在,清晰了吧?
那怎么append时,从0开始呢???
这不是很简单,直接让第二个参数为0。
var names = make([]int,0,10)//结果:[11 23 231]如图所示。
好了,这个,懂了吧,怎么继续哈。
为什么不推荐使用var []类型方式创建切片
我们上述一直在提一个词,自动扩容。
我们来看这样一段普通的代码。
package main
import "fmt"
func main() { var names []int //地址:0x0,长度(len):0,容量(cap):0 fmt.Printf("地址:%p,长度(len):%d,容量(cap):%d\n", names, len(names), cap(names)) names = append(names, 1, 2, 3)
//地址:0xc000010380,长度(len):3,容量(cap):4 fmt.Printf("地址:%p,长度(len):%d,容量(cap):%d\n", names, len(names), cap(names))}虽然按照这种方法,使用append动态添加是没问题的。
在不使用make声明数组时,len和cap都是0,并且地址也是一个值。
通过append之后,可以明显看到,地址发生了改变,因为又重新申请了数组,切片重新指向新的数组。
len和cap也发生了变化。
copy复制切片
package main
import "fmt"
func main() { var names1 = make([]string, 0, 10) names1 = append(names1, "张三") names1 = append(names1, "李四") var names2 = names1 //将names1赋值到names2 fmt.Println(names1, names2) //[张三 李四] [张三 李四] names1[0] = "张三666"//修改names下标为0的值为 张三666 fmt.Println(names1, names2) //[张三666 李四] [张三666 李四] //为什么修改names1的值,会影响names2的值????}为什么修改names1的值,会影响names2的值???
这个,就又要回到内存分布图了,如图所示。
我们说过很多次,不管是打印,还是赋值等操作,只会操作栈上面存储的值。
当names2=names1时,只会把names1栈上面的地址,给names2。
但是存的时堆上面的地址,终究还是指向了同一个堆。
所以修改names1时,names2也修改了。
那如果不想出现上述问题怎么办???
解决办法:使用copy
package main
import "fmt"
func main() { var names1 = make([]string, 0, 10) names1 = append(names1, "张三") names1 = append(names1, "李四") //定义一个names2切片用于接收,第二个参数要留空间,names1里面又几个元素,names2第二个参数也要是几 var names2 = make([]string, 2, 10) copy(names2, names1)//将names1的值,赋值到names2 fmt.Println(names1, names2) //[张三 李四] [张三 李四] names1[0] = "张三666"//修改names下标为0的值为 张三666 fmt.Println(names1, names2) //[张三666 李四] [张三 李四] fmt.Printf("names1地址:%p names2地址:%p\n",names1,names2) //names1地址:0xc00009a0a0 names2地址:0xc00009a140}内存图
自动扩容机制
非常抱歉,我不会
。。。
总结
上述我们学习了Go基础之切片补充。如果在操作过程中有任务问题,记得在下面的讨论区留言,我们看到会第一时间解决问题。
我是码农星期八,如果觉得还不错,记得动手点赞一下哈。感谢你的观看。
------------------- End -------------------
欢迎大家点赞,转发,转载,感谢大家的相伴与支持
本站仅提供存储服务,所有内容均由用户发布,如发现有害或侵权内容,请点击举报
打开APP,阅读全文并永久保存 查看更多类似文章
猜你喜欢
类似文章
【热】打开小程序,算一算2024你的财运
Go 语言系列11:切片
详解 Go 的切片
Go slice切片详解和实战:make append copy
go:内置函数 | 闭包 | 数组 | 切片 | 排序 | map | 锁
golang 使用 strings.Split 切割的注意
GO蛋疼的学习笔记 | 似水流年
更多类似文章 >>
生活服务
热点新闻
分享 收藏 导长图 关注 下载文章
绑定账号成功
后续可登录账号畅享VIP特权!
如果VIP功能使用有故障,
可点击这里联系客服!

联系客服