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,会 panicp++、p + 1 等非法),所以别想用它遍历数组——该用切片就用切片新手常问“为什么传结构体要加 *”,答案不在语法,而在语义和开销:
*int、*Person,因为 Go 默认按值传递,传值会复制一份struct,传值可能拷贝上百字节;传 *MyStruct 只传 8 字节地址(64 位系统)func (p *Person) SetName(...)),那调用方必须用指针实例,否则编译报错:cannot call pointer method on p
package main import "fmt"type Config struct { Timeout int Debug boo
l }
func updateTimeout(c *Config, t int) { c.Timeout = t // 修改原始结构体 }
func main() { cfg := Config{Timeout: 30} updateTimeout(&cfg, 60) fmt.Println(cfg.Timeout) // 输出 60 }
Go 里很多类型自带“间接性”,但和指针无关:
[]int、map[string]int、chan int 是引用类型,本身已包含底层指针字段(如 slice 的 data ptr),传值时复制的是 header,不是整个底层数组或哈希表——不需要、也不该再套一层 *[]int
*[4]int 是指向数组的指针,[]int 是切片;二者内存布局完全不同,(*[4]int)(unsafe.Pointer(&arr)) 这类转换极危险,且通常没必要interface{} 能装任何值,但它不是指针类型;装小值(如 int)时直接存值,装大结构体时才可能存指针——这是 runtime 内部优化,你不用管这不是理论风险,是每天都在发生的运行时错误。只要函数参数或返回值可能为 nil,就必须显式检查:
m["key"] 返回 *Value,但 key 不存在时是 nil)nil 的构造函数(如 json.Unmarshal 传入 *T,但 T 未初始化)*string 类型,但 JSON 解析时该字段缺失 → 字段为 nil
正确写法永远是:if p != nil { use(*p) },而不是赌它“应该不为空”。
指针在 Go 里没有魔法,它的价值恰恰在于透明:& 就是取地址,* 就是读内存,nil 就是空。把“能不能”换成“要不要”,把“怎么写对”换成“为什么必须这么写”,路就清楚了。