




该用指针时:需修改原值、避免大结构体拷贝(>128字节)、实现接口、与C交互或需地址唯一性;不该用时:纯计算输入、只读配置初始化、返回新值不改原数据;特别注意sync.Mutex不可取地址后传。
Go 中指针不是为了“性能优化”而存在的默认选项。值类型(int、string、struct 小于机器字长时)传参开销极小,盲目加 * 反而增加 nil panic 风险和内存逃逸概率。
struct)、实现接口(如 io.Reader 要求方法集在指针上)、与 C 交互或需地址唯一性func max(a, b int) int)、配置结构体只读初始化、返回新值不改原数据(如 func WithTimeout(cfg Config) Config)sync.Mutex 不能取地址再传——mu := &sync.Mutex{} 是错的;必须保证其地址不变,所以定义时就该是字段或全局变量字段是否加 * 取决于语义而非大小。如果字段代表“可选”或“可为空”,才用指针;否则优先用值类型,让零值有意义且安全。
type User struct {
ID int // 值类型:ID 永远有默认 0,不会 nil
Name string // 值类型:空字符串 "" 合法且明确
Role *string // 指针类型:仅当 Role 可能“未设置”(区别于 "")才需要
}
*string 能表达三种状态:nil(未设)、""(设为空)、"admin"(设为值),但代价是每次访问都要判空string;若必须区分,再用指针,但应在文档或字段名中体现意图(如 RoleOpt *string)Go 函数参数是传值,但传的是指针的副本——这常被误解为“传引用”。真正危险的是把局部变量地址返回,或对切片底层数组做意外修改。
func NewConfig() *Config {
c := Config{Version: "1.0"}
return &c // c 是栈上局部变量,返回其地址导致悬垂指针(实际 Go 编译器会自动逃逸,但语义已混乱)
}func NewConfig() Config {
return Config{Version: "1.0"}
}
// 使用时:
cfg := NewConfig()
process(&cfg) // 显式控制生命周期
cap 或重分配底层数组:传 []byte 进函数后做 append,可能让调用方看到意外扩容——如需隔离,显式 copy 一份用 interface{} 接收值后,再取地址会丢失原始类型信息;泛型约束中混用指针和值类型会导

var x int = 42
val := interface{}(x)
ptr := &val // ptr 类型是 *interface{},不是 *int,无法直接转回type T any,传 *int 和 int 是不同类型,T 不会自动解引用最常被忽略的一点:Go 的 GC 不回收“逻辑上已失效但仍有指针指向”的内存。过度使用指针延长对象生命周期,反而造成内存堆积——尤其在缓存、池化、事件回调中,记得及时置 nil 或用 sync.Pool 管理。