与数组的区别
数组是固定长度的,而分片确实可动态增长的,以定义为例:
// 定义数组, 一定要指定长度var names [5]string// 定义分片, 不需要指定长度var names []string
在函数调用时, 数组是值传递,而分片是引用传递
其实对于 golang
来讲,函数调用的时候都是值传递,拷贝一个副本, 之所以表现为值传递和引用传递,在于一个拷贝的是数据值,另一个拷贝的是数据指针,两个指针值指向的是同一个内存地址。
分片的实现
分片的底层数据还是使用的数组,它一共包含 3 个字段:
地址指针
长度
容量
// source 是一个分片, 大小为 4, 容量为: 5source := make([]string, 4, 5) fmt.Println(source, len(source), cap(source))// 输出: [ ] 4 5// 注意这里不会进行内存分配, 因为 source 还有剩余空间可以新加数据source = append(source, "1") fmt.Println(source, len(source), cap(source))// 输出: [ 1] 5 5
在使用 append
来为分片添加数据时, 是否有内存分配很重要
当 append
没有内存分配时:
source := []string{"1", "2", "3", "4", "5"}// 拷贝 source 的第二到第三个元素(不包括第三个)// copied 的容量包括: 3, 4,5copied := source[2:3] fmt.Println("source-->", source) fmt.Println("copied-->", copied) fmt.Println("接下来修改 copied 的内容,看是否会对 source 产生影响...")// 这里 append 内部不会创建一个新的底层数组,共有 source 的底层数组// 因为 copied 的容量足够新加一个元素// 所以不会影响到 source 的内容copied = append(copied, "mike") fmt.Println("source-->", source) fmt.Println("copied-->", copied)
输出:
source--> [1 2 3 4 5]copied--> [3]接下来修改 copied 的内容,看是否会对 source 产生影响...source--> [1 2 3 mike 5]copied--> [3 mike]
当 append
有内存分配时:
source := []string{"1", "2", "3", "4", "5"}// 拷贝 source 的第二到第三个元素(不包括第三个)// copied 的容量包括: 3copied := source[2:3:3]// 此时 copied 会和 source 共享底层数组fmt.Println("source-->", source) fmt.Println("copied-->", copied) fmt.Println("接下来修改 copied 的内容,看是否会对 source 产生影响...")// 这里 append 内部会创建一个新的底层数组,不会共有 source 的底层数组// 所以不会影响到 source 的内容copied = append(copied, "mike") fmt.Println("source-->", source) fmt.Println("copied-->", copied)
输出:
source--> [1 2 3 4 5] copied--> [3] 接下来修改 copied 的内容,看是否会对 source 产生影响... source--> [1 2 3 4 5] copied--> [3 mike]
作者:痕无落
链接:https://www.jianshu.com/p/3d6aaa1e9a90
点击查看更多内容
为 TA 点赞
评论
共同学习,写下你的评论
评论加载中...
作者其他优质文章
正在加载中
感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦