Golang 方法接收者为值与指针的区别

Golang 中同时有函数和方法。方法是一个包含了接收者(receiver)的函数,receiver可以是内置类型或者自定义类型struct的一个值或者是一个指针。所有给定类型的方法组成该类型的方法集。

方法的定义语法格式如下:

func (r ReceiverType) funcName(parameters) (return values){
    //body
}

如下面的这个例子,定义了一个新类型Integer,它和int一样,只是为它增加了个新方法Less()。

 

package main
import "fmt"
type Integer int 

func (a Integer) Less(b Integer) bool {
    return a < b 
}

func main() {
    var a Integer = 1 

    if a.Less(2) {
        fmt.Println("less then 2")
    }   
}

 

 

./hello 
less then 2

 

可以看出,Golang 在自定义类型的对象中没有C++/Java那种隐藏的this指针,而是在定义成员方法时显式声明了其所属的对象。

方法接收者为对象的指针与值有什么区别呢?如果方法接收者为对象的指针,则会修改原对象,如果方法接收者为对象的值,那么在方法中被操作的是原对象的副本,不会影响原对象。示例如下:

 

 
package main  
  
import "fmt"  

type Integer int 
 
//乘2 
func (p *Integer) double() int {   
    *p = *p * 2   
    fmt.Printf("double p = %d\n", *p)  
    return 0   
}  

//平方  
func (p Integer) square() int {   
    p = p * p   
    fmt.Printf("square p = %d\n", p)  
    return 0   
}  
  
func main() {
    var i Integer = 2   
    i.double()                //receiver 为对象的指针,原对象被修改
    fmt.Println("i = ", i)  
    
    i.square()                //receiver 为对象的值,原对象不会被修改
    fmt.Println("i = ", i)  
}
 

 

root@ubuntu:~/go_learn/example.com/hello# ./hello 
double p = 4
i =  4
square p = 16
i =  4

 

使用方法时注意以下几点:

(1)不管方法的 receiver 是对象的值还是指针,对象的值和指针均可以调用该方法。即对象的值既可以调用 receiver 是值的方法,也可以调用 receiver 是指针的方法。对象的指针也是如此;

(2)当方法的接收者是值时,即使是指针调用,那么方法内部也是对原对象的副本进行操作,不会影响原对象;

(3)当方法的接收者是指针时,即使用值调用,那么方法内部也是通过指针对原对象进行操作,会影响原对象。

 

posted on 2021-07-19 18:07  tycoon3  阅读(145)  评论(0编辑  收藏  举报

导航