Go语言Slice的一道骚题

大家好我是一根甜苦瓜,今天我们来聊一道go语言中slice的面试题,加深一下对slice的理解

题目来自于雨痕大佬的《Go学习笔记》第四版,P43页

go 复制代码
func TestDemo(t *testing.T) {
	slice := []int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
	s1 := slice[2:5]     
	s2 := s1[2:6:7]      
	s2 = append(s2, 100)
	s2 = append(s2, 200)
	s1[2] = 20           
	t.Log(s1)            
	t.Log(s2)        
	t.Log(slice)
}

暂停5分钟,大家可以人脑编译一下,看看结果回事什么

时间到,公布答案

go 复制代码
=== RUN   TestDemo
    user_repo_test.go:21: [2 3 20]
    user_repo_test.go:22: [4 5 6 7 100 200]
    user_repo_test.go:23: [0 1 2 3 20 5 6 7 100 9]
--- PASS: TestDemo (0.00s)
PASS

是不是脑瓜子嗡嗡的呀?下面来图解这道题

先来看初始化 slices1s2 这三行代码:

● 第 22 行将 slice 初始化为一个数组,元素是 0-9 这没有什么好说的。

● 第 23 行,是在 slice 的基础上创建的 s1,选择下标范围 [2,5) 的元素来创建。左闭右开。容量为 3,cap 为 8

● 第 24 行,s2s1 的基础上创建,下标范围是 s1[2,6)cap 容量到了 7 号下标,容量为 4cap5

所以最终的三个对象的内存布局如下图:

接下来看代码s2 = append(s2, 100),由于 s2cap5,此时尚有一个空间可以插入,所以直接 append。但是由于共用了底层数组 s1slice,所以最终,他们三个变量都会被影响,最终结果如下:

接下来看代码s2 = append(s2, 200),此时 s2lencap 都为 5,那么在s2上继续添加元素显然不够用了,需要扩 容,扩容是新开辟了空间,将老数据复制过去,所以s2和老数组并无瓜葛了。这里是直接翻倍扩容(关于扩容大家可以去百度一下)。至此三个变量的内容如下。

最后再看这行代码 s1[2] = 20

这个就很简单了,此时 s1 还是和老数组有瓜葛的,所以一改全改但是不会影响到新开辟的 s2。最终三个数组/切片的情况就是下面这个图。

这里需要注意的是,由于打印 S1 的时候并不会把 len 以外的数据打印出来,所以最终输出结果是:

相关推荐
缺点内向21 分钟前
C#编程实战:如何为Word文档添加背景色或背景图片
开发语言·c#·自动化·word·.net
一起养小猫30 分钟前
Flutter for OpenHarmony 实战:记账应用数据统计与可视化
开发语言·jvm·数据库·flutter·信息可视化·harmonyos
韩立学长39 分钟前
【开题答辩实录分享】以《智能大学宿舍管理系统的设计与实现》为例进行选题答辩实录分享
数据库·spring boot·后端
zhougl99640 分钟前
Java 所有关键字及规范分类
java·开发语言
java1234_小锋1 小时前
Java高频面试题:MyISAM索引与InnoDB索引的区别?
java·开发语言
2501_944525541 小时前
Flutter for OpenHarmony 个人理财管理App实战 - 支出分析页面
android·开发语言·前端·javascript·flutter
qq_417129251 小时前
C++中的桥接模式变体
开发语言·c++·算法
开源技术2 小时前
如何将本地LLM模型与Ollama和Python集成
开发语言·python
Hello World . .2 小时前
数据结构:队列
c语言·开发语言·数据结构·vim
clever1012 小时前
在QtCreator 4.10.2中调试qt程序qDebug()输出中文为乱码问题的解决
开发语言·qt