指针数组是元素为指针的数组,如[3]int;数组指针是指向整个数组的指针,如[3]int。二者类型、内存布局、使用方式及语义均不同,不可混用。

指针数组是指一个数组,它的每个元素都是指针;数组指针是指一个指针,它指向的是整个数组(而非单个元素)。二者类型、内存布局、使用方式和语义完全不同,不能混用。
类型写法与本质区别
Go 中类型声明顺序从右向左读,这是理解的关键:
-
指针数组:如
[]*int—— “一个切片,元素类型是 *int”,即每个元素都是指向 int 的指针。实际常用的是切片(动态),但若用数组则是[3]*int(长度为 3 的数组,每个元素是 *int)。 -
数组指针:如
*[3]int—— “一个指针,指向一个长度为 3 的 int 数组”。它不指向某个 int,而是指向整个 [3]int 这块连续内存的起始地址。
内存与访问方式不同
假设定义:
arr := [3]int{10, 20, 30}
ptrArr := [3]*int{&arr[0], &arr[1], &arr[2]}
arrPtr := &arr
-
ptrArr占用至少 3 个指针大小的内存(如 24 字节),存储三个独立地址,可分别解引用获取值:*ptrArr[0]得 10。 -
arrPtr是单个指针(8 字节),指向arr起始地址;要访问元素需用(*arrPtr)[1](先解引用得数组,再下标取值)。
常见使用场景
- 指针数组(或 []*T 切片):适合需要“持有多个变量地址”并可能单独修改其值的场景。例如缓存中保存一组结构体指针、回调列表、树节点子节点指针集合等。
-
数组指针(*[N]T):较少直接使用,多见于函数参数传递,避免大数组值拷贝。例如
func processBigArray(a *[10000]int)比func processBigArray(a [10000]int)高效得多——前者传地址,后者传副本(10000×int 大小)。
容易混淆的点
- Go 没有“数组引用”类型,
*[N]T是真指针,支持unsafe.Sizeof验证它是 8 字节(64 位系统)。 -
[]T是切片(头结构,含指针+长度+容量),不是数组;*[]T是指向切片头的指针,和本文讨论的两种类型都不同。 - 声明时别写错:
*[3]int≠[3]*int,Go 编译器会严格检查类型,混用直接报错。
基本上就这些。记牢“指针在前还是在后”,再结合用途想一想:是要存一堆地址?还是只想高效传一个大数组?选对类型,代码更准也更稳。










