Go中指针变量var声明时默认值为nil,是明确零值;解引用前须判空;初始化常用&取地址、new()分配零值内存、或&Struct{…}复合字面量。

Go 中 var 声明指针变量时默认值是 nil
Go 的指针类型变量在仅用 var 声明时,不指向任何地址,其值为 nil。这不是“未初始化”,而是明确的零值 —— 和 int 是 0、string 是 "" 一样自然。
常见误解是以为没赋值就“危险”或“随机”,其实安全,但直接解引用会 panic:
var p *int fmt.Println(*p) // panic: runtime error: invalid memory address or nil pointer dereference
-
var p *int→p == nil,合法,可安全比较或传参 - 必须先让它指向有效内存,才能读写
*p - 函数参数接收
*T类型时,也常接受nil,需主动判空
用 & 取地址完成初始化最常用
绝大多数场景下,指针初始化就是取某个已有变量的地址。这是最直观、最无歧义的方式:
age := 25 p := &age // p 是 *int 类型,指向 age 所在内存
- 右侧必须是“可寻址的值”:变量、结构体字段、切片元素等;
&3、&(x + y)都非法 -
:=推导出指针类型,不用写*int;若用var p *int = &age也完全等价 - 注意生命周期:不能返回局部变量地址给调用方(编译器会逃逸分析并自动分配到堆),但初学者不必手动干预
用 new() 函数分配零值内存
new(T) 返回一个指向新分配的、类型为 T 的零值的指针,即 *T。它不调用构造函数,也不支持带参数初始化:
立即学习“go语言免费学习笔记(深入)”;
p := new(int) // 等价于:var v int; p := &v → *p == 0 s := new(string) // *s == ""
-
new(T)本质是:分配内存 + 写入零值 + 返回地址,一步到位 - 无法用于需要非零初始值的场景,比如想让
*int指向42,只能先new(int)再赋值*p = 42 - 对结构体有用但易被误用:
new(Struct)得到所有字段为零值的实例,但更推荐字面量 + 取地址:p := &Struct{Field: 1}
用复合字面量 &Struct{...} 初始化结构体指针
这是初始化结构体指针最推荐的方式,兼顾可读性、灵活性和效率:
type User struct {
Name string
Age int
}
u := &User{Name: "Alice", Age: 30} // 直接获得 *User
- 字段可选填,未写的字段自动为零值(
Name为空字符串,Age为0) - 支持嵌套结构体、切片字面量、甚至内联函数调用结果
- 与
new(User)+ 逐字段赋值相比,语义更紧凑,且编译器优化更好 - 注意:如果结构体含不可比较字段(如
map、func),仍可通过此方式初始化,但后续不能用==比较指针所指内容
nil 指针当普通变量用,忘了在解引用前加判空;或者误以为 new(T) 能执行初始化逻辑——它只做零值填充。










