贝利信息

如何在Golang中理解切片和指针关系_底层数组共享机制

日期:2026-01-04 00:00 / 作者:P粉602998670
切片不是指针但含指向底层数组的指针,共享行为源于该指针相同且内存重叠;传参是值传递但指针仍有效;修改元素会影响共享数组,扩容则切断共享;可用copy或append(nil, src...)创建独立底层数组。

切片不是指针,但它内部包含一个指向底层数组的指针——这个指针才是共享行为的根源。理解这点,就能解释为什么修改一个切片会影响另一个,也能预判何时影响会消失。

切片结构体里藏着一个指针

Go 中的切片本质是一个三字段结构体:

你写 s := []int{1,2,3},运行时会分配一段数组内存,并让 s.array 指向它首地址;后续所有截取、赋值、传参,只要没扩容,这个指针值基本不变。

共享发生于指针相同、内存重叠

两个切片是否相互影响,取决于它们的 array 字段是否指向同一块内存,且修改位置落在彼此可访问范围内。

reflect.ValueOf(s).Pointer() 可验证指针值是否一致,但更关键的是看索引是否交叉重叠。

函数传参时,切片本身是值传递,但指针仍有效

把切片传进函数,复制的是那个三字段结构体,其中的 array 指针也被复制了——新副本依然指向原数组。

切断共享:主动创建独立底层数组

不想被意外修改?不能靠“不改”,而要主动隔离内存。

不复杂但容易忽略。