在 go 中编写方法时,关键决策之一是是按值还是按指针传递结构体。此选择会影响性能、代码行为和内存分配。在这篇文章中,我们将通过实际示例探讨这种差异,并了解每种方法何时更合适。
让我们从一个小结构体和两个方法开始:一个结构体通过值传递,另一个通过指针传递。
package main import ( "fmt" ) type person struct { name string age int } // method with struct passed by value func (p person) celebratebirthdayvalue() { p.age++ } // method with struct passed by pointer func (p *person) celebratebirthdaypointer() { p.age++ } func main() { person := person{name: "alice", age: 30} // passing by value person.celebratebirthdayvalue() fmt.println("after celebratebirthdayvalue:", person.age) // output: 30 // passing by pointer person.celebratebirthdaypointer() fmt.println("after celebratebirthdaypointer:", person.age) // output: 31 }
值和指针之间的区别
当我们将结构体按值传递给方法时,go 会创建该结构体的副本。对方法内的结构体进行的任何更改都不会影响原始结构体,因为我们正在使用独立的副本。
另一方面,当我们通过指针传递结构体时,我们传递的是原始结构体的内存地址。这意味着对方法内的结构体所做的任何更改都将直接修改原始结构体,因为我们正在操作同一个实例。
总结:
-
按值: 该方法接收结构体的副本,创建一个新的内存空间。
-
通过指针:该方法接收原始结构体的内存地址,指向同一内存空间。
堆
当结构按值传递时,副本会在堆栈上分配,这通常是快速且高效的。但是,如果结构很大,则副本可能会消耗大量堆栈内存。
当通过指针传递结构时,指针本身会在堆栈上分配,但原始结构可能会在堆上分配,特别是当它是使用 new、make 创建或由匿名函数捕获时。
堆分配在分配时间和垃圾收集方面更昂贵,但允许高效操作大量数据,而无需复制整个结构。
何时使用每种方法
按价值
按值传递结构在以下情况下很有用:
- 您要确保原始结构不被修改。
- 结构体很小,复制它不会显着影响性能。
- 该方法只需读取数据,无需更改内部状态。
示例:
func (p person) getname() string { return p.name }
这里,getname 只读取 name 字段并返回一个字符串,而不修改结构体的状态。
通过指针
在以下情况下通过指针传递结构是有益的:
- 需要修改原来的结构体。
- 该结构体很大,复制其数据在内存和性能方面的成本很高。
- 您希望避免不必要的副本以提高代码效率。
示例:
func (p *Person) UpdateName(newName string) { p.Name = newName }
在这种情况下,updatename 直接修改原始结构体的 name 字段,这比创建副本更高效。
结论
在 go 中编写方法时决定是按值还是按指针传递结构体是一个重要的选择,可能会影响性能、代码行为和内存分配。
按值传递对于确保方法内结构的不变性很有用,而按指针传递对于修改原始结构并在处理较大结构时优化性能至关重要。