Go切片是值类型,底层由ptr、len、cap三字段构成;赋值时复制结构体,因ptr共享而表现类似引用;append扩容时若len==cap会分配新数组,导致ptr改变、脱离原底层数组。
Go语言中切片(slice)常被说成“引用类型”,但严格来说它本身是值类型,只是其底层数据结构包含指向底层数组的指针——这使得它在行为上表现得像引用类型。理解这一点,关键要看它的底层结构和扩容机制。
每个切片变量实际是一个结构体,包含三个字段:
这个结构体本身是值类型,赋值或传参时会复制这三个字段。但由于ptr是共享的,所以多个切片可能指向同一块底层数组——这是“类似引用”行为的根源。
当两个切片共用同一底层数组(比如通过切片操作生成),它们的ptr指向相同内存地址。此时修改其中一个切片的某个元素,就是在改那块内存里的值,另一个切片自然读到变化后的结果。
例如:
a := []int{1,2,3,4}调用append时,如果原切片的len ,直接在底层数组空闲位置写入,不改变ptr;一旦len == cap,就要扩容:
注意:扩容后的新切片p
tr已变,与原切片不再共享底层数组,后续修改互不影响。
不要误以为“切片是引用类型所以传参不用加&”——它确实是值传递,只是值里含指针。也不必手动管理内存,但要注意:
make([]T, len, cap)
copy另存一份释放引用)reflect.DeepEqual或逐元素比基本上就这些。搞懂ptr、len、cap三者关系,再结合扩容规则,切片的行为就清晰了。