指针传递地址避免复制,适合大结构体和修改需求;值传递复制数据,适用于小型对象和只读操作。

Go语言中,指针与值类型的使用在语法和语义上存在明显差异,理解这些差异对写出高效、安全的代码至关重要。核心区别在于数据传递方式和内存操作行为:值类型传递副本,指针传递地址,从而影响函数调用、结构体方法定义以及内存效率。
函数参数中的值与指针传递
当将变量传入函数时,Go默认采用值传递。对于值类型(如int、string、struct),这意味着会复制整个数据。如果结构体较大,开销显著。
使用指针可以避免复制,直接操作原始数据。例如:
- 值传递:函数接收的是结构体副本,修改不影响原对象
- 指针对传递:函数通过地址访问原始结构体,可直接修改其字段
方法接收者的选择:值 vs 指针
定义方法时,接收者可以是值类型或指针类型。选择不同会影响方法能否修改实例状态。
立即学习“go语言免费学习笔记(深入)”;
- 值接收者:方法内部操作的是实例的副本,适合只读操作
- 指针接收者:方法可修改调用者本身,适用于变更字段或维持一致性
结构体字段中的指针使用
结构体字段可以声明为指针类型,用于共享数据或表示可选值(nil 表示缺失)。
例如,一个用户信息结构体中,地址可能是可选的:
- *Address 类型字段允许赋 nil,节省内存且表达语义清晰
- 直接嵌入值类型则必须初始化,无法表示“无”状态
同时,多个结构体指向同一块内存可减少冗余,但也需注意并发修改风险。
性能与安全的权衡
虽然指针避免了复制开销,但过度使用可能导致内存逃逸、GC压力上升和调试困难。
- 小型结构体或基本类型通常无需指针,值传递更高效
- 频繁在堆上分配指针对象可能拖慢程序
- nil 指针解引用会引发 panic,需做好判空处理
基本上就这些。掌握指针与值的适用场景,能让Go代码更清晰、高效。不复杂但容易忽略。










