Go语言允许用户定义类型。当用户声明一个类型时,这个声明就给编译器提供了一个框架,告知必要的内存大小和信息。
声明结构类型
type user struct {
name string
email string
}
上述代码声明的结构类型有2个字段,每个字段都基于一个内置类型。一旦声明了类型,就可以使用这个类型创建值。
使用结构类型声明变量
var zuckjet user
我们用关键字var创建了类型为user且名为zuckjet的变量。当声明变量时,这个变量对应的值总是会被初始化。这个值要么是用指定的值初始化,要么用零值来初始化。对数值来类型来说,零值是0;对字符串来说,零值是空字符串。
下面我们来看一下如何声明一个user类型的变量,并使用某个非零值作为初始值。
zuckberg := user{
name: "zuckberg"
email: "zuckjet@gmail.com"
}
方法
方法能给用户定义的类型添加新的行为。方法实际上也是函数,知识在声明的时,在关键字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", "zuckjet@gmail.com"}
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", "zuckjet@gmail.com"}
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", "zuckjet@gmail.com"}
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", "zuckjet@gmail.com"}
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 zuckjet@gmail.com}
&{zuckjet zuckjet@gmail.com}
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。