Go指针是显式内存地址操作工具,非引用传递替代品;*T表示指向T的地址,&x仅对可寻址变量取址,nil解引用会panic,且不支持指针算术。

Go 里的指针不是“传引用”的替代品,而是明确的内存地址操作工具——它不隐藏行为,也不自动解引用,理解这点,就避开了 80% 的新手困惑。
什么是 *T 和 &?别从 C 类比
Go 指针类型写作 *int、*string,表示“指向一个 int 值的地址”,不是“int 的引用”。&x 是唯一合法获取地址的方式,且只接受可寻址变量(比如 var x int),不能对字面量(如 &42)或表达式(如 &x + 1)取地址,否则编译失败。
-
var p *int声明后,p的值是nil,不是随机地址 -
*p是解引用操作,类型就是int;若p == nil时执行*p,会 panic - Go 不支持指针算术(
p++、p + 1等非法),所以别想用它遍历数组——该用切片就用切片
什么时候必须用指针?看三个真实场景
新手常问“为什么传结构体要加 *”,答案不在语法,而在语义和开销:
-
修改原变量:函数内想改外部变量,必须传
*int、*Person,因为 Go 默认按值传递,传值会复制一份 -
避免大结构体拷贝:一个含 10 个字段的
struct,传值可能拷贝上百字节;传*MyStruct只传 8 字节地址(64 位系统) -
方法接收者一致性:如果结构体方法用了指针接收者(
func (p *Person) SetName(...)),那调用方必须用指针实例,否则编译报错:cannot call pointer method on p
package main import "fmt"type Config struct { Timeout int Debug bool }
func updateTimeout(c *Config, t int) { c.Timeout = t // 修改原始结构体 }
func main() { cfg := Config{Timeout: 30} updateTimeout(&cfg, 60) fmt.Println(cfg.Timeout) // 输出 60 }
哪些“看起来像指针”的东西其实不是?警惕常见混淆
Go 里很多类型自带“间接性”,但和指针无关:
立即学习“go语言免费学习笔记(深入)”;
-
[]int、map[string]int、chan int是引用类型,本身已包含底层指针字段(如 slice 的 data ptr),传值时复制的是 header,不是整个底层数组或哈希表——不需要、也不该再套一层*[]int -
*[4]int是指向数组的指针,[]int是切片;二者内存布局完全不同,(*[4]int)(unsafe.Pointer(&arr))这类转换极危险,且通常没必要 -
interface{}能装任何值,但它不是指针类型;装小值(如int)时直接存值,装大结构体时才可能存指针——这是 runtime 内部优化,你不用管
新手最容易 panic 的地方:nil 指针解引用
这不是理论风险,是每天都在发生的运行时错误。只要函数参数或返回值可能为 nil,就必须显式检查:
- 从 map 中取值后得到的指针(如
m["key"]返回*Value,但 key 不存在时是nil) - 调用可能返回
nil的构造函数(如json.Unmarshal传入*T,但 T 未初始化) - 结构体字段是
*string类型,但 JSON 解析时该字段缺失 → 字段为nil
正确写法永远是:if p != nil { use(*p) },而不是赌它“应该不为空”。
指针在 Go 里没有魔法,它的价值恰恰在于透明:& 就是取地址,* 就是读内存,nil 就是空。把“能不能”换成“要不要”,把“怎么写对”换成“为什么必须这么写”,路就清楚了。










