最近踩了一个go的坑,而这个坑的根本原因就是slice和arry的区别理解不清楚导致的。
一、问题提出,slice在被复制之后,修改复制之后的那个slice内容,结果复制之前的slice内容也变化了。
我们先来看一段代码:
package main
import (
"fmt"
"reflect"
)
func main() {
var arr [8]int = [8]int{1}
var arr1 [8]int
var slice, slice1 []int
fmt.Println("type of:", reflect.TypeOf(arr), reflect.TypeOf(slice))
arr1 = arr
fmt.Println("1: arr|arr1 of:", arr, arr1)
arr1 = [8]int{3}
fmt.Println("2: arr|arr1 of:", arr, arr1)
slice = append(slice, 2)
slice1 = slice
fmt.Println("3: slice|slice1 of:", slice, slice1)
slice1[0]= 5
fmt.Println("4: slice|slice1 of:", slice, slice1)
}
output:
type of: [8]int []int
1: arr|arr1 of: [1 0 0 0 0 0 0 0] [1 0 0 0 0 0 0 0]
2: arr|arr1 of: [1 0 0 0 0 0 0 0] [3 0 0 0 0 0 0 0] // 只是修改的那个变化了
3: slice|slice1 of: [2] [2]
4: slice|slice1 of: [5] [5] // 好奇怪,怎么都更改了呢
结果分析:
看到输出结果,笔者开始是蒙的,怎么两种[8]int,[]int的复制会有两个结果呢?arry的复制之后,两个arry都是独立的,slice的复制之后,内容却会同时变化。
二、问题原因
开始之前,我们先看下什么是slice,什么是arry?
例子:
var arr [8]int // 这种指定了长度的数组就是arry类型,复制都是值的复制一份。
var slice []int // 这种 就是slice类型,更像是指针,复制操作其实就是操作指针的公共地址。
Go中将arry和slice当作了两种数据类型,一个是与int等基本类型具有相同的特征,一个是引用类型与指针和interface具有相同的特征。也正是这个区别,导致了上面代码里面的操作,输出结果会有这种不同。
这种问题,当发生在[8]int赋值给[:]int的时候,会潜移默化的将arry类型转换成slice类型,而我们一旦后续修改了赋值之后的数据,就会踩这个坑,[8]int内存中的数据也被改掉了。
灰子学技术:https://mp.weixin.qq.com/s/zeseVMGDPuLH-00T1v7dlQ
有疑问加站长微信联系(非本文作者))