指针数组和切片结合可高效管理内存。定义指针数组如var ptrArr [3]int,存变量地址并解引用访问值;切片更灵活,如var ptrSlice []int,用append添加指针,遍历时*ptr读取值;切片操作共享底层数组,子切片指针仍指向原地址,修改影响所有引用;适用场景包括减少大结构体复制、函数间共享修改数据,需注意避免循环中取地址覆盖和悬空指针问题。

在Go语言中,指针数组和切片是常见且灵活的数据结构。它们可以结合使用来高效管理内存和数据引用。理解如何操作指针数组以及对切片进行处理,有助于写出更高效、更安全的代码。
指针数组的定义与初始化
指针数组是指数组中的每个元素都是指向某种类型的指针。例如,一个指向整数的指7针数组可以这样声明:
var ptrArr [3]*int a, b, c := 10, 20, 30 ptrArr[0] = &a ptrArr[1] = &b ptrArr[2] = &c
此时,ptrArr 是一个长度为3的数组,每个元素都是 *int 类型,保存的是变量的地址。通过 *ptrArr[i] 可以访问其指向的值。
切片中的指针操作
切片比数组更常用,因为它具有动态长度。你可以创建一个指向整数的指针切片:
立即学习“go语言免费学习笔记(深入)”;
var ptrSlice []*int a, b, c := 10, 20, 30 ptrSlice = append(ptrSlice, &a, &b, &c)
现在 ptrSlice 包含三个指向整数变量的指针。遍历时可通过解引用来读取或修改值:
for _, ptr := range ptrSlice {
fmt.Println(*ptr) // 输出 10, 20, 30
}
切片操作对指针的影响
切片操作如 slice[i:j] 返回原切片的子切片,共享底层数组。如果原切片包含指针,子切片中的指针仍指向相同的地址。
这意味着修改通过指针指向的值,会影响所有引用该地址的切片或变量:
a := 42
ptrs := []*int{&a}
subPtrs := ptrs[0:1]
*subPtrs[0] = 99 // 修改通过子切片的指针
fmt.Println(a) // 输出 99
这种行为在处理大量数据时非常高效,但也需注意避免意外修改。
常见使用场景与建议
- 当结构体较大时,使用 []*Struct 可避免复制开销。
- 在函数间传递切片时,若需修改原始数据,使用指针切片可直接操作原值。
- 注意指针生命周期,避免悬空指针(虽然Go有GC,但逻辑错误仍可能发生)。
- 初始化局部变量后再取地址存入切片,防止循环中取地址覆盖问题:
var ptrs []*int
for i := 0; i < 3; i++ {
val := i
ptrs = append(ptrs, &val) // 正确:每次创建新变量
}
如果不引入 val,直接用 &i,所有指针会指向同一个循环变量,最终值可能异常。
基本上就这些。指针数组和切片在Go中处理得自然且高效,关键是理解指针语义和切片的共享机制。合理使用能提升性能,也能避免常见陷阱。










