Go中数组是值类型,修改原数组必须传入指向数组的指针([N]T);通过ptr解引用后遍历并索引修改,如(ptr)[i] = 2;切片虽可修改底层数组但语义不同,不绑定具体数组。

在 Go 语言中,数组是值类型,直接传递或赋值会复制整个数组;而指针可以让我们通过地址操作原始数据,实现“通过引用修改数组元素”的效果。关键在于:要修改原数组,必须传入指向数组的指针(如 *[3]int),而非数组本身或切片。
理解 Go 中数组的值语义
Go 的数组是固定长度、值类型。这意味着:
- 赋值
a := b(其中 a、b 是同类型数组)会复制所有元素; - 函数传参时,
func f(arr [3]int)接收的是副本,修改不影响原数组; - 只有显式使用指针(
*[N]T)才能真正修改调用方的数组内存。
使用数组指针遍历并修改元素
声明指向数组的指针后,需用 *ptr 解引用得到原数组,再通过索引修改:
func doubleArray(ptr *[3]int) {
for i := range *ptr {
(*ptr)[i] *= 2
}
}
func main() {
arr := [3]int{1, 2, 3}
doubleArray(&arr) // 传入地址
fmt.Println(arr) // 输出 [2 4 6]
}
注意:range *ptr 遍历的是解引用后的数组;(*ptr)[i] 是必须加括号的,因为 *ptr[i] 会被解析为 *(ptr[i])(错误)。
立即学习“go语言免费学习笔记(深入)”;
对比:切片方式更常用但语义不同
虽然切片([]int)也能“修改底层数组”,但它不是数组指针:
- 切片本身是结构体(含指针、长度、容量),传参时复制的是这个结构体,但其内部指针仍指向原底层数组;
- 因此
func f(s []int) { s[0] = 99 }能修改原底层数组内容(前提是未扩容); - 但切片无法保证操作的是某个具体数组变量(例如长度不匹配、扩容后指向新内存),而
*[N]T是严格绑定到特定数组的。
安全遍历指针数组的常见写法
为避免硬编码长度,可结合 len 和类型推导:
func incrementAll(ptr *[5]int) {
n := len(*ptr)
for i := 0; i < n; i++ {
(*ptr)[i]++
}
}也可用 for i := range *ptr,它自动基于数组长度迭代,更简洁且不易越界。
不复杂但容易忽略:Go 没有隐式引用传递,想改原数组,& 和 * 缺一不可。










