Go基础编程:方法(method)

1.基本概念

Go不像其它面相对象语言一样可以写个class,然后在class里面写一堆方法,但是它也很巧妙的实现了这种效果,我们只需要在普通函数前面加个接受者(receiver,写在函数名前面的括号里面),这样编译器就知道这个函数(方法)属于哪个struct了

method是附属在一个给定的类型上,语法和函数的声明语法几乎一样,只是再func后面增加了一个recevier(也就是method所依从的主体)

2.method定义

func (r ReceiverType) funcName(parameters) (results)

形象一点说,就是 ReceiverType 类型的所有字段,方法 funcName 都是可以使用的,可以认为 funcName 属于 ReceiverType

package main
import (
    "fmt"
    "math"
)
type Rectangle struct {
    width, height float64
}
type Circle struct {
    radius float64
}
func (r Rectangle) area() float64 {
    return r.width * r.height
}
func (c Circle) area() float64 {
    return c.radius * c.radius * math.Pi
}
func main() {
    r1 := Rectangle{12, 2}
    r2 := Rectangle{9, 4}
    c1 := Circle{10}
    c2 := Circle{25}
    fmt.Println("Area of r1 is: ", r1.area())//Area of r1 is:  24
    fmt.Println("Area of r2 is: ", r2.area())//Area of r2 is:  36
    fmt.Println("Area of c1 is: ", c1.area())//Area of c1 is:  314.1592653589793
    fmt.Println("Area of c2 is: ", c2.area())//Area of c2 is:  1963.4954084936207
}

method 是通过 . 来访问,就像访问struct里面字段一样。

method 里面可以访问接受者的字段,比如 r1.area() 就可以访问 r1 里面的 width 和 height。

虽然 method 的名字是一样的,但是不同的 receiver 不一样,那么 method 就不一样。这一点很重要哦

还有一点,method不仅能作用再struct上,也可以定义再任何自定义的类型、内置类型等各种类型上面。

method 中的 receiver 可以是值传递,也可以是指针。指针的话,就可以直接修改 receiver 中的内容。

3.method特性

  • Go中虽没有class,但依旧有method
  • 通过显示说明receiver来实现与某个类型的组合
  • 只能为同一个包中的类型定义方法
  • Receiver可以是类型的值或者指针
  • 不存在方法重载
  • 可以使用值或指针来调用方法,编译器会自动完成转换
  • 从某种意义上来说,方法是函数的语法糖,因为receiver其实就是方法所接收的第1个参数)
  • 如果外部结构和嵌入结构存在同名方法,则优先调用外部结构的方法
  • 类型别名不会拥有底层类型所附带的方法
  • 方法可以调用结构中的非公开字段

举例:

package main
import "fmt"  
type A struct {
    Name string
}

type B struct {
    Name string
}

func main() {
    a := A{}
    a.Print()
    fmt.Println(a.Name)

    b := B{}
    b.Print()
    fmt.Println(b.Name)
}

// 指针传递
func (a *A) Print() {
    a.Name = "AA"
    fmt.Println("A")
}

func (b B) Print() {
    b.Name = "BB"
    fmt.Println("B")
}

打印:

➜  myfirstgo go run method.go
A
AA
B

➜  myfirstgo

其他类型的方法绑定

type TZ int

func main() {
    var a TZ
    a.Print()
    fmt.Println(a)
}

func (a *TZ) Print() {
    fmt.Println("TZ")
}

打印:

➜  myfirstgo go run method.go
TZ
0

最后说下访问权限,因为Go是以大小写来区分是公有还是私有,但都是针对包级别的,所以在包内所有的都能访问,而方法绑定本身只能绑定包内的类型,所以方法可以访问接收者所有成员。如果是包外调用某类型的方法,则需要看方法名是大写还是小写,大写能被包外访问,小写只能被包内访问。

發表回覆

你的電郵地址並不會被公開。 必要欄位標記為 *