今天刷题leetcode.78子集,其中遇到一个语法坑,查了半小时最后才定位是切片的使用问题,现在来总结一下,直接上代码。
func Test_Slice(t *testing.T) {
s1 := []int{1, 2, 3, 4}
fmt.Println(s1) // [1 2 3 4]
s2 := s1[:2]
fmt.Println(s2) // [1 2]
s2 = append(s2, 10)
fmt.Println(s2) // [1 2 10]
fmt.Println(s1) // ?
}
最后s1的值是多少呢?
[1 2 10 4]
为啥会这麽奇怪呢?
原因是在执行完 s2 := s1[:2] 这句时,s1和s2其实指向的是同一个底层数组(有空大家可以去看一下golang源码声明切片的那个struct)
两者的区别只是指针限定了左右界限不同,
因此对s2的操作也会影响s1的值。
同样的,append如果没有发生扩容(len < cap),也会存在这个问题,需要注意下:
func Test_Slice2(t *testing.T) {
s1 := make([]int, 0, 10) //容量是10
s1 = append(s1, 1)
s1 = append(s1, 2) // s1=[1, 2]
s2 := append(s1, 3) // s2=[1, 2, 3]
s1 = append(s1, 10)
fmt.Println(s1) // s1=[1 2 10]
fmt.Println(s2) // s2=?
}
s2的值也是 [1 2 10]!
因为s1和s2指向的是同一个底层数组,s1把数组的第三个值改为10,同时也影响了s2的值。
怎么解决上面的问题呢?其实只要复制一份切片,让两个切片别指向同一个底层数组就行了。
引申:拷贝一个切片的最佳实践
// 复制切片
func CopySlice() {
nums := []int{1, 2, 3}
newNums := make([]int, len(nums))
copy(newNums, nums)
}