登录
首页 >  Golang >  Go教程

Golang数组指针定义与使用详解

时间:2026-03-11 19:38:35 376浏览 收藏

本文深入解析了Go语言中固定长度数组指针(*[N]T)的核心概念与实战要点,强调其与切片([]T)、元素指针(*T)及切片指针(*[]T)的本质区别:*[N]T是直接指向连续内存中N个T类型元素的指针,不可扩容、不携带长度信息,且N必须为编译期常量;它在避免数组值拷贝、保障Cgo内存安全互通(尤其对接C的T(*)[N]参数)时不可或缺,但极易因误用&arr[0]、临时字面量取址、类型混淆等导致编译失败或运行时崩溃——掌握正确声明(var p *[3]int)、赋值(p = &arr)、解引用((*p)[i])和生命周期约束,是写出高效、健壮、跨语言兼容Go代码的关键。

如何在Golang中创建一个指向数组的指针_*[N]T

怎么声明 *[N]T 类型的指针

Go 中数组是值类型,*[N]T 是指向固定长度数组的指针,不是切片。它和 []T*[]T 完全不同,不能自动扩容,也不能用 len/cap 直接操作指针本身(得先解引用)。

声明方式很简单:

var p *[3]int
arr := [3]int{1, 2, 3}
p = &arr

注意:&arr 得到的就是 *[3]int,不是 *int;别写成 &arr[0]——那只是首元素地址,类型是 *int,丢了长度信息。

  • *[N]T 的 N 必须是编译期常量,不能是变量或 const 以外的表达式
  • 声明但未赋值时,pnil,解引用会 panic
  • 不能用 new([N]T) 直接得到 *[N]T?可以,但等价于 &[N]T{},语义略绕,建议直接取址

为什么不能用 *[]T 替代 *[N]T

常见误解:以为“指针+切片”能模拟固定数组指针。其实不行——*[]T 是指向切片头的指针,切片头包含指针、长度、容量三个字段,跟数组内存布局不兼容。

典型错误场景:

func takesPtrToArray(p *[2]string) { /* ... */ }
slice := []string{"a", "b"}
// ❌ 编译错误:cannot use &slice (type *[]string) as type *[2]string
takesPtrToArray(&slice)
// ✅ 正确做法:必须基于真实数组
arr := [2]string{"a", "b"}
takesPtrToArray(&arr)
  • *[N]T*[]T 类型不兼容,无法隐式转换
  • Cgo 场景下尤其关键:C 函数期望 T(*)[N],只有 Go 的 *[N]T 能安全传入;*[]T 传过去会崩溃或读错内存
  • 函数参数用 *[N]T 可避免复制整个数组,但调用方必须有对应长度的数组变量

*[N]T 解引用后怎么用

解引用得到的是原数组,支持所有数组操作:索引、遍历、传给接受 [N]T 的函数。但要注意,*p 是值,不是引用——修改 (*p)[i] 会影响原数组,因为 p 指向它。

arr := [2]int{10, 20}
p := &arr
(*p)[0] = 99 // arr 现在是 [99, 20]
fmt.Println(arr) // [99 20]
  • 别写 p[0] —— 这是非法语法,*[N]T 不支持直接索引
  • 想遍历?必须先解引用:for i, v := range *p { ... }
  • 传给需要 [N]T 的函数时,直接传 *p(即解引用后的值),不是 p

Cgo 场景下传 *[N]T 到 C 函数的坑

C 函数若声明为 void foo(int (*arr)[5]),Go 端必须用 *[5]int,且数组必须是变量(不能是字面量临时值,否则取地址可能悬空)。

// ✅ 安全
arr := [5]int{1,2,3,4,5}
C.foo((*C.int)(unsafe.Pointer(&arr))) // 注意类型转换
<p>// ❌ 危险:&[5]int{...} 取的是临时数组地址,生命周期只在当前语句
C.foo((*C.int)(unsafe.Pointer(&[5]int{1,2,3,4,5})))</p>
  • Go 数组在内存中是连续 N×sizeof(T) 字节,和 C 完全一致,这是能互通的基础
  • 务必确保 Go 数组变量的生命周期长于 C 函数调用——局部变量没问题,但不要传闭包捕获的、可能被提前回收的数组
  • 如果 C 接口是 int * + 长度参数,那用 &arr[0] 更合适,此时不需要 *[N]T

*[N]T 看似简单,真正容易出问题的地方在于:它要求调用链每一环都严格匹配长度,且必须基于命名的数组变量——临时数组字面量、切片、类型别名混用,都会在编译期或运行期悄悄掉坑里。

文中关于的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《Golang数组指针定义与使用详解》文章吧,也可关注golang学习网公众号了解相关技术文章。

资料下载
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>