在go语言中,使用指针接收者修改结构体能提升性能;这是因为指针接收者不会复制结构体本身,而是复制一个地址,代价很小;相比之下,值接收者会复制整个结构体,带来较大的开销;尤其当结构体较大或方法频繁调用时,这种差异更加明显;基准测试显示,指针接收者的性能显著优于值接收者;因此,在需要修改结构体内容时应优先使用指针接收者,而仅在读取小结构体数据时可考虑值接收者;不确定结构体大小时,默认使用指针接收者更安全;尽管编译器有时会优化,但理解并主动选择合适的接收者类型对性能至关重要。

在 Go 语言中,使用指针来修改结构体的方法是非常常见的做法。很多人知道“应该用指针接收者”,但未必清楚为什么这么做更好,尤其是在性能方面。其实核心原因就一个:避免结构体复制带来的开销。

指针接收者 vs 值接收者
Go 中的方法可以定义在结构体的值类型或指针类型上:
值接收者:方法操作的是结构体的副本。指针接收者:方法操作的是原始结构体的引用。
当你调用一个值接收者方法时,Go 会自动复制整个结构体。如果结构体很大,这个复制动作就会带来不小的性能损耗。
立即学习“go语言免费学习笔记(深入)”;
为什么用指针能提升性能?
简单来说:指针接收者不会复制结构体本身,而是复制一个地址(通常是 4 或 8 字节),无论结构体多大,代价都很小。

比如下面这个结构体:
type User struct { ID int Name string Age int}
即使它只有几个字段,每次调用值接收者方法时都会复制整个 User。如果你的方法只是读取数据,这可能没问题;但如果方法频繁被调用、结构体又很大,这种复制就会成为瓶颈。
基准测试对比性能差距
我们来做个简单的基准测试,比较值接收者和指针接收者在修改结构体字段时的性能差异。
测试代码如下:
package mainimport "testing"type User struct { ID int Name string Age int}// 值接收者方法func (u User) SetNameVal(name string) { u.Name = name}// 指针接收者方法func (u *User) SetNamePtr(name string) { u.Name = name}func BenchmarkSetNameVal(b *testing.B) { u := User{} for i := 0; i < b.N; i++ { u.SetNameVal("Alice") }}func BenchmarkSetNamePtr(b *testing.B) { u := &User{} for i := 0; i < b.N; i++ { u.SetNamePtr("Alice") }}
执行结果(示例):
BenchmarkSetNameVal-8 1000000000 0.35 ns/opBenchmarkSetNamePtr-8 2000000000 0.15 ns/op
可以看到,在这个简单例子中,指针接收者的性能明显优于值接收者。虽然单次差异不大,但在高并发场景下累积起来是可观的。
实际开发中的建议
如果你需要修改结构体内容,请优先使用指针接收者。如果你只读结构体且结构体较小,使用值接收者也可以接受。如果你不确定结构体会有多大,默认用指针接收者更安全。某些情况下,即使你不显式传指针,Go 编译器也会做优化(比如自动取址),但这不意味着你可以忽略性能差异。
小贴士:有时候你可能会看到这样的写法:func (u User) GetSomething() string { … }func (u *User) SetSomething(v string) { … }这样混合使用值和指针接收者是可以的,但要注意一致性,尤其是涉及到接口实现的时候。
基本上就这些。指针在方法中修改结构体的优势不仅体现在语义清晰,也实实在在地影响着程序的性能表现。
以上就是Golang指针在方法修改结构体时的优势 通过基准测试展示性能差距的详细内容,更多请关注创想鸟其它相关文章!
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1390911.html
微信扫一扫
支付宝扫一扫