Golang指针数组与数组指针有什么区别_Golang两者差异与适用场景

指针数组是元素为指针的数组,如[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 编译器会严格检查类型,混用直接报错。

基本上就这些。记牢“指针在前还是在后”,再结合用途想一想:是要存一堆地址?还是只想高效传一个大数组?选对类型,代码更准也更稳。