周游列国之仕子 2019-06-28
Go语言允许用户定义类型。当用户声明一个类型时,这个声明就给编译器提供了一个框架,告知必要的内存大小和信息。
type user struct { name string email string }
上述代码声明的结构类型有2个字段,每个字段都基于一个内置类型。一旦声明了类型,就可以使用这个类型创建值。
var zuckjet user
我们用关键字var创建了类型为user且名为zuckjet的变量。当声明变量时,这个变量对应的值总是会被初始化。这个值要么是用指定的值初始化,要么用零值来初始化。对数值来类型来说,零值是0;对字符串来说,零值是空字符串。
下面我们来看一下如何声明一个user类型的变量,并使用某个非零值作为初始值。
zuckberg := user{ name: "zuckberg" email: "[email protected]" }
方法能给用户定义的类型添加新的行为。方法实际上也是函数,知识在声明的时,在关键字func和方法名之间增加了一个参数。
func (u user) notify() { fmt.Printf(u.name) }
在上面的代码中,关键字func和函数名之间的参数被称作接收者,将函数与接收者的类型绑在一起。如果一个函数有接收者,这个函数就被称为方法。下面我们来看一个简单的demo:
package main import ( "fmt" ) type user struct { name string email string } func (u user) notify() { fmt.Println(u.name) } func main() { bill := user{"zuckjet", "[email protected]"} bill.notify() } // zuckjet
首先我们顶一个一个user类型,然后notify使用值接收者实现了一个方法,最后我们通过创建一个user类型的变量bill并调用notify方法。
当我们通过bill.notify()调用方法的时候,notify方法中的u变量的值就等同于变量bill。为什么我说等同于变量bill呢?因为
使用值接收者声明方法,调用时会使用这个值的一个副本来执行。
我们稍微修改一下上面的代码即可验证:
package main import ( "fmt" ) type user struct { name string email string } func (u user) notify() { u.name = "zuckberg" fmt.Println(u.name) } func main() { bill := user{"zuckjet", "[email protected]"} bill.notify() fmt.Println(bill.name) } // zuckberg // zuckjet
因为变量u是bill的一个副本,所以改变它里面属性的值,是不会影响bill本身的值的。
值得注意的是,在Go语言中我们还可以使用指针来调用值接收者声明的方法,尽管这看起来似乎不太合理。
我们看一下下面的代码:
package main import ( "fmt" ) type user struct { name string email string } func (u user) notify() { fmt.Println(u.name) } func main() { bill := &user{"zuckjet", "[email protected]"} bill.notify() } // zuckjet
这次我们定义的变量bill是指针类型的,用指针类型的变量来调用noify方法的时候,依然能够输出bill的name属性值。看到这里或许你有些疑惑,不是说方法notify中的变量u就等同于变量bill吗,为啥bill是一个指针类型的时候还能够正确输出name属性值呢。其实,我们可以认为Go在背后替我们做了如下操作:
(*bill).notify()
上面我们讲述的是使用值接受者声明方法,除此之外我们还可以使用指针接收者声明方法。我们来看下面的代码:
package main import ( "fmt" ) type user struct { name string email string } func (u *user) notify() { fmt.Println(u.name) } func main() { bill := user{"zuckjet", "[email protected]"} bill.notify() } // zuckjet
同样在调用notify方法时,Go在背后做了这样一件事情:
(&bill).notify()
当把上面的notify()方法改造为:
func (u *user) notify() { fmt.Println(u.name) fmt.Println(&u) fmt.Println(*u) fmt.Println(u) } // 输出结果 zuckjet 0xc420088018 {zuckjet [email protected]} &{zuckjet [email protected]}