指针数组是[N]*T,每个元素为指向变量的指针;数组指针是*[N]T,指向整个数组的指针,二者类型和用途不同。

在Golang中,指针数组和数组指针是两个容易混淆但本质不同的概念。理解它们的关键在于分析类型声明的结构和内存布局。
指针数组(Array of Pointers)
指针数组是指一个数组,其每个元素都是指针类型。也就是说,数组本身存储的是多个指向其他变量的地址。
声明方式:var arr [N]*T,其中 N 是数组长度,T 是被指向的类型。
示例:
立即学习“go语言免费学习笔记(深入)”;
var ptrArr [3]*int a, b, c := 10, 20, 30 ptrArr[0] = &a ptrArr[1] = &b ptrArr[2] = &c
此时,ptrArr 是一个长度为3的数组,每个元素是一个 *int 类型的指针,分别指向变量 a、b、c 的地址。
特点:
- 数组大小固定,元素是独立的指针
- 可以部分赋值 nil
- 修改某个指针不会影响其他元素
数组指针(Pointer to Array)
数组指针是指向整个数组的指针。它本身是一个指针,指向一块连续的数组内存。
声明方式:var ptr *[N]T,表示 ptr 是指向长度为 N、类型为 T 的数组的指针。
示例:
立即学习“go语言免费学习笔记(深入)”;
arr := [3]int{1, 2, 3}
var arrPtr *[3]int = &arr
fmt.Println(arrPtr) // 输出地址
fmt.Println(*arrPtr) // 输出 [1 2 3]
这里 arrPtr 指向整个 [3]int 类型的数组,通过 *arrPtr 可以访问原数组内容。
特点:
- 它是指针,占用固定指针大小(如8字节)
- 可用于函数传参避免拷贝大数组
- 类型必须匹配数组长度和元素类型
实际使用场景对比
假设要处理一组整数:
- 用指针数组:适合需要动态管理多个独立变量地址的场景,比如缓存某些变量的引用
- 用数组指针:适合传递大数组给函数,避免值拷贝带来的性能损耗
函数参数中的典型应用:
func processArrayPtr(p *[3]int) {
fmt.Println("接收到数组:", *p)
}
func processPtrArray(pa [3]*int) {
for i, v := range pa {
if v != nil {
fmt.Printf("索引 %d 的值是 %d\n", i, *v)
}
}
}
常见误区与注意事项
- 切片不是数组指针:虽然切片底层有指针,但它包含长度和容量,是更复杂的数据结构
- 数组长度是类型的一部分:*[3]int 和 *[4]int 是不同类型,不能互相赋值
- 指针数组初始化时元素默认为 nil,解引用前需确保已赋值有效地址
- 使用 new([N]T) 可创建数组指针并返回其地址
基本上就这些。关键看类型表达式的层次:[N]*T 是多个指针组成的数组,*[N]T 是一个指向数组的单一指针。理解这一点,使用时就不会混淆了。










