go语言 指针类型的方法接受者和非指针类型的方法接受者之间的区别
近来写了一些go语言程序,对go语言的操作也没有觉得特别有问题,今天因为一个手误,将本来该写成指针的方法,写成了非指针,一直没有深入思考过这个问题。于是查了一些资料,做一个简单的记录。
本文主要记录了go语言 指针类型的方法接受者和非指针类型的方法接受者之间的区别,以及这两种写法自己的理解。
go语言 指针类型的方法接受者和非指针类型的方法接受者之间的区别
先看一个非常常见的例子,很多讲解 指针方法接受者和非指针类型的方法接受者之间区别的文章都举的例子。
指针类型的方法接受者:
package main
import "fmt"
type Person struct {
name string
age int
}
func (v *Person) changeName(name string) {
p := v
p.name = name
}
func main() {
xiaoMing := Person{name: "lilei", age: 33}
xiaoMing.changeName("hanmeimei")
fmt.Println(xiaoMing.name)
}
输出结果为:
hanmeimei
以上写法没有问题,结果也是希望看到的。
非指针类型的方法接受者
package main
import "fmt"
type Person struct {
name string
age int
}
//就这里把 * 好去掉了。
func (v Person) changeName(name string) {
p := v
p.name = name
}
func main() {
xiaoMing := Person{name: "lilei", age: 33}
xiaoMing.changeName("hanmeimei")
fmt.Println(xiaoMing.name)
}
输出结果为:
lilei
从以上代码的运行结果可以得出如下结论:
1. 指针类型的方法接受者:能够改变原先结构体的数值
2. 非指针类型的方法接受者:不能够改变原来结构体的数值。
上面的说法个人的理解为:
在调用Persion的对象 xiaoMing 上的方法modifyName() 的时候,表面上我们仅仅传递了一个参数 “hanmeimei”。 其实传递了两个参数,一个隐藏的参数就是 xiaoMing本身。 有点类似C++ 中的this,或者说叫做上下文,或者调用者本身。
方法的接受者为指针
func (v *Person) changeName(name string) {
p := v
p.name = name
}
//方法接受者不是指针的时候
func (v Person) changeName(name string) {
p := v
p.name = name
}
既然传递了两个参数一个是 参数列表中的 name 字段。另外一个就是 xiaoMing本身,将xiaoMing对象传递给 上面的 方法接受者 v 。
如果方法接受者是指针,则传递过去的时候,是同一个对象的引用,也就是将xiaoMing的指针赋值给了v v 和xiaoMing是同一个对象,所以在方法内部修改属性的值,属性的值是发生改变的
如果方法接受者不是指针,则传递过去的时候,是该对象xiaoMing的一个copy,在内部修改了对应的属性,和xiaoMing对象已经不是同一个对象了,因此在输出xiaoMing.name 的时候,该值并没有发生变化。
很赞哦! (2)