GoLang:数组的切片失效

GoLang:数组的切片失效

学习切片的时候想到一个问题:如果一个切片是引用一个完整的数组,那么给这个切片 append 一个元素后,这个 append 的元素会以什么样的形式存在?

先看代码:

	arr := []int{1, 2, 3, 4, 5, 6, 7, 8, 9}
	s := arr[:]
	s[8] = 666
	fmt.Println(arr)
	fmt.Println(s)
	fmt.Printf("&arr[8] = %p, &s[8] = %p\n", &arr[8], &s[8])


	// a := 10
	// fmt.Println(&a)

	s = append(s, 10)
	fmt.Println(s)

	fmt.Printf("&arr[8] = %p, &s[8] = %p, &s[9] = %p\n", &arr[8], &s[8], &s[9])

	s[8] = 921
	fmt.Println(arr)
	fmt.Println(s)

当切片 s 去引用整个数组 arr 时,改变切片就是改变数组,这个都清楚。

可以如果我现在给这个切片 s ,append 一个元素 10,那么这个 10 将会存在哪里呢?切片是否还是数组的引用呢?

所以我做个这个小实验

结论:

在这里插入图片描述

当切片引用整个数组时,它的长度和容量就是数组的长度。所以当我 append 的时候,会发现容量不够,会发生扩容操作。

所以我推断,当发生扩容时,会在堆上开辟一段连续的空间将数组完整的拷贝过来(因为此时的数组中的元素存在栈上),再加上追加的元素。所以我们看到后来 arr[8] 和 s[8] 的地址都已经不一样了。之后再操作切片就跟数组没关系了。

叮~?

Golang数组切片是两种不同的数据类型,用于存储相同数据类型的容器。数组的长度是固定的,而切片的长度是可变的。在日常应用中,切片的使用更为普遍。 数组在声明时需要指定长度,并且在初始化时必须提供相同长度的元素。例如,`a := int{1, 2, 3}`就是一个长度为3的整数数组数组的长度一旦确定后就不能更改。 切片是基于数组的引用类型。它不需要指定固定的长度,并且可以根据需要动态扩展或缩小。切片包装着底层数组,通过指定起始索引和结束索引来指定子集。例如,`b := a[:]`就是一个切片,它包含了数组a的所有元素。 数组适用于需要固定长度的场景,而切片适用于长度可变的情况。在实际应用中,切片更常用,因为它提供了更大的灵活性和便利性。 总结: - 数组是长度固定的容器,切片是长度可变的容器; - 数组在声明时需要指定长度,切片则不需要; - 数组的长度一旦确定后就不能更改,而切片可以根据需要动态扩展或缩小; - 切片是基于数组的引用类型,可以通过指定起始索引和结束索引来指定子集。 参考资料: Golang中的「数组」和「切片」都是存储同一数据类型的容器,只不过Golang中的数组长度是固定的,而切片的长度是可变化的。 切片是引用类型,切片包装的数组称为该切片的底层数组。我们来看一段代码://a是一个数组,注意数组是一个固定长度的,初始化时候必须要指定长度,不指定长度的话就是切片了 a := int{1, 2, 3} //b是数组,是a...。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值