Golang method详解

  从严格意义上讲,Go语言不算一门面向对象的编程语言,至少没有提供关键字class,没有明确类的概念,更没有明确封装、继承、重载、多态等面向对象的概念。但这并不影响其面向对象的能力,取而代之的是其定义的struct概念,与C/C++中的struct不同,Go中的structtype特性承载了其所有面向对象的功能,本文中我们来看一下Go中method功能。

自定义类型

  在提及方法之前,先看一下Go中对自定义类型的支持,类似与C/C++中的typedef,Go中使用type关键字定义自定义类型。typeName可以是一个包或者函数内唯一的任何合法的Go标识符。typeSpecification可以是任何内置的类型(如string、int、切片、映射或者通道)、接口、结构体或者一个函数签名。

type typeName typeSpecification

  此处首先提出自定义类型,主要是为了下文铺垫,Go语言中规定,方法是作用在自定义类型的值上的一类特殊函数,通常自定义类型的值会被传递给该函数。该值可以以指针或者值的形式传递,这取决于方法如何定义。

方法

  如上文所述,方法只能作用在自定义类型的值,这其中有两层含义:

  • 自定义类型:必须是自定义类型,而不能是Go内建类型
  • 值:只能作用在对象上,而不能作用在自定义类型上
      上述内容只限定了方法的作用范畴,还没有提到方法到底是什么呢。在Go中,方法其实就是函数,只不过此函数和自定义类型的值绑定在一起,其定义方式和函数几乎类似,只是增加了一个限定描述符,即除了需要在 func 关键字和方法名之间必须写上接收者(写入括号中)之外,该接收者既可以以该方法所属于的类型的形式出现,也可以以一个变量名及类型的形式出现。
      可以为任何自定义类型添加一个或者多个方法,方法的接收者总是一个该类型的值,或者只是该类型值的指针。然而,对于任何一个给定的类型,每个方法名必须唯一。唯一名字的要求是,
  • 不能同时定义两个相同名字的方法,让其中一个的接收者为指针类型而另一个为值类型。
  • 不支持重载方法,也就是说,不能定义名字相同但是不同签名的方法。一种提供等价方法的方式是使用可变参数,但Go语言推荐的方式是使用名字唯一的函数。

举例

package employee
  
import  "fmt"
  
type Employee struct {  
  
    FirstName   string
    LastName    string
    TotalLeaves int
    LeavesTaken int
}
  
func (e Employee) LeavesRemaining() {  
    fmt.Printf("%s %s has %d leaves remaining", e.FirstName, e.LastName, (e.TotalLeaves - e.LeavesTaken))
}

  上述代码中定义了一个结构体Employee,结构体中有一个方法LeaveRemaining这个方法计算并且显示剩余Employee的数目。然后可以调用上述方法。

package main
  
import "employee"
 
func main() {  
    e := employee.Employee {
        FirstName: "Sam",
        LastName: "Adolf",
        TotalLeaves: 30,
        LeavesTaken: 20,
    }
    e.LeavesRemaining()
}

运行结果如下

Sam Adolf has 10 leaves remaining  

模拟构造函数的例子

  Go不支持构造函数,编译器不接受结构体为空值的情况,此时可以通过一个支持的函数NewT(parameters) 来完成,这个实现类似C++中类的构造函数。如果一个包只定义了一种类型,那么可以通过New(parameters) 来代替NewT(parameters)。

package employee
 
 
import (  
    "fmt"
)
 
 
type employee struct {  
    firstName   string
    lastName    string
    totalLeaves int
    leavesTaken int
}
  
func New(firstName string, lastName string, totalLeave int, leavesTaken int) employee {  
    e := employee {firstName, lastName, totalLeave, leavesTaken}
    return e
}
  
func (e employee) LeavesRemaining() {  
    fmt.Printf("%s %s has %d leaves remaining", e.firstName, e.lastName, (e.totalLeaves - e.leavesTaken))
}

  在这里有一个重要的改变,通过结构体名首字母改为小写e,结构体名称从Employee变为employee,也就是将employee结构体限制为不可导出状态,从而阻止其他的包调用。通常设置一个不可导出的结构体所有成员都不能被导出是一个好的习惯(注意到上面的结构体各成员的首字母也变成小写了,如果一个结构体类型的名称以大写字母开头,则该结构体被导出,其他包可以访问它。同样地,如果结构体中的字段名以大写字母开头,则这些字段也可以被其他包访问)。其调用方式也需要同步修改。

package main  
 
import "oop/employee"
 
func main() {  
    e := employee.New("Sam", "Adolf", 30, 20)
    e.LeavesRemaining()
}

程序输入为

Sam Adolf has 10 leaves remaining 

method特性

封装特性

  如上文提到,Go区分公有属性和私有属性的机制就是方法或属性是否首字母大写,如果首字母大写的方法就是公有的,如果首字母小写的话就是私有的。此处不再举例说明。

继承特性

  Go中继承方式采用的是匿名组合的方式,如下述代码所示,Woman 结构体中包含匿名字段Person,那么Person中的属性也就属于Woman对象。

package main

import "fmt"

type Person struct {
    name string
}

type Woman struct {
    Person
    sex string
}

func main() {
    woman := Woman{Person{"wangwu"}, "女"}
    fmt.Println(woman.name)
    fmt.Println(woman.sex)
}

多态特性

package main

import "fmt"

type Eater interface {
    Eat()
}

type Man struct {
}

type Woman struct {
}

func (man *Man) Eat() {
    fmt.Println("Man Eat")
}

func (woman *Woman) Eat() {
    fmt.Println("Woman Eat")
}

func main() {
    var e Eater

    woman := Woman{}
    man := Man{}

    e = &woman
    e.Eat()

    e = &man
    e.Eat()
}

  Go中的多态功能其实是通过interface功能实现的,下一节再对其做详细介绍。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,529评论 5 475
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,015评论 2 379
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,409评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,385评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,387评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,466评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,880评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,528评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,727评论 1 295
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,528评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,602评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,302评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,873评论 3 306
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,890评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,132评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,777评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,310评论 2 342

推荐阅读更多精彩内容