11 接口(Interfaces)与反射(reflection)

11.1 接口是什么

Go 语言不是一种“传统”的面向对象编程语言:它里面没有类和继承的概念。

但是 Go 语言里有非常灵活的接口概念,通过它可以实现很多面向对象的特性。接口提供了一种方式来说明对象的行为:如果谁能搞定这件事,它就可以用在这儿。

接口定义了一组方法(方法集),但是这些方法不包含(实现)代码:它们没有被实现(它们是抽象的)。接口里也不能包含变量。

通过如下格式定义接口:

typeNamerinterface{Method1(param_list) return_typeMethod2(param_list) return_type    ...}

上面的Namer是一个接口类型

(按照约定,只包含一个方法的)接口的名字由方法名加[e]r后缀组成,例如Printer、Reader、Writer、Logger、Converter等等。还有一些不常用的方式(当后缀er不合适时),比如Recoverable,此时接口名以able结尾,或者以I开头(像.NET或Java中那样)。

Go 语言中的接口都很简短,通常它们会包含 0 个、最多 3 个方法。

不像大多数面向对象编程语言,在 Go 语言中接口可以有值,一个接口类型的变量或一个接口值:var ai Namer,ai是一个多字(multiword)数据结构,它的值是nil。它本质上是一个指针,虽然不完全是一回事。指向接口值的指针是非法的,它们不仅一点用也没有,还会导致代码错误。

此处的方法指针表是通过运行时反射能力构建的。

类型(比如结构体)实现接口方法集中的方法,每一个方法的实现说明了此方法是如何作用于该类型的:即实现接口,同时方法集也构成了该类型的接口。实现了Namer接口类型的变量可以赋值给ai(接收者值),此时方法表中的指针会指向被实现的接口方法。当然如果另一个类型(也实现了该接口)的变量被赋值给ai,这二者(译者注:指针和方法实现)也会随之改变。

类型不需要显式声明它实现了某个接口:接口被隐式地实现。多个类型可以实现同一个接口

实现某个接口的类型(除了实现接口方法外)可以有其他的方法

一个类型可以实现多个接口

接口类型可以包含一个实例的引用, 该实例的类型实现了此接口(接口是动态类型)

即使接口在类型之后才定义,二者处于不同的包中,被单独编译:只要类型实现了接口中的方法,它就实现了此接口。

所有这些特性使得接口具有很大的灵活性。

第一个例子:

示例 11.1interfaces.go

packagemainimport"fmt"typeShaperinterface{Area()float32}typeSquarestruct{    sidefloat32}func(sq*Square)Area()float32{returnsq.side* sq.side}funcmain() {sq1:=new(Square)    sq1.side=5//var areaIntf Shaper//areaIntf = sq1//shorter,without separate declaration://areaIntf := Shaper(sq1)//or even:areaIntf:=sq1    fmt.Printf("The square has area:%f\n", areaIntf.Area())}

输出:

The square has area: 25.000000

上面的程序定义了一个结构体Square和一个接口Shaper,接口有一个方法Area()。

在main()方法中创建了一个Square的实例。在主程序外边定义了一个接收者类型是Square方法的Area(),用来计算正方形的面积:结构体Square实现了接口Shaper。

所以可以将一个Square类型的变量赋值给一个接口类型的变量:areaIntf = sq1。

现在接口变量包含一个指向Square变量的引用,通过它可以调用Square上的方法Area()。当然也可以直接在Square的实例上调用此方法,但是在接口实例上调用此方法更令人兴奋,它使此方法更具有一般性。接口变量里包含了接收者实例的值和指向对应方法表的指针。

这是多态的 Go 版本,多态是面向对象编程中一个广为人知的概念:根据当前的类型选择正确的方法,或者说:同一种类型在不同的实例上似乎表现出不同的行为。

如果Square没有实现Area()方法,编译器将会给出清晰的错误信息:

cannot use sq1 (type *Square) as type Shaper in assignment:

*Square does not implement Shaper (missing Area method)

如果Shaper有另外一个方法Perimeter(),但是Square没有实现它,即使没有人在Square实例上调用这个方法,编译器也会给出上面同样的错误。

扩展一下上面的例子,类型Rectangle也实现了Shaper接口。接着创建一个Shaper类型的数组,迭代它的每一个元素并在上面调用Area()方法,以此来展示多态行为:

示例 11.2interfaces_poly.go

packagemainimport"fmt"typeShaperinterface{Area()float32}typeSquarestruct{    sidefloat32}func(sq*Square)Area()float32{returnsq.side* sq.side}typeRectanglestruct{    length, widthfloat32}func(rRectangle)Area()float32{returnr.length* r.width}funcmain() {r:=Rectangle{5,3}//Area() of Rectangle needs a valueq:=&Square{5}//Area() of Square needs a pointer//shapes := []Shaper{Shaper(r), Shaper(q)}//or shortershapes:=[]Shaper{r, q}    fmt.Println("Looping through shapes for area ...")forn,_:=rangeshapes {        fmt.Println("Shape details:", shapes[n])        fmt.Println("Area of this shape is:", shapes[n].Area())    }}

输出:

Looping through shapes for area ...

Shape details:  {5 3}

Area of this shape is:  15

Shape details:  &{5}

Area of this shape is:  25

在调用shapes[n].Area())这个时,只知道shapes[n]是一个Shaper对象,最后它摇身一变成为了一个Square或Rectangle对象,并且表现出了相对应的行为。

也许从现在开始你将看到通过接口如何产生更干净更简单更具有扩展性的代码。在 11.12.3 中将看到在开发中为类型添加新的接口是多么的容易。

下面是一个更具体的例子:有两个类型stockPosition和car,它们都有一个getValue()方法,我们可以定义一个具有此方法的接口valuable。接着定义一个使用valuable类型作为参数的函数showValue(),所有实现了valuable接口的类型都可以用这个函数。

示例 11.3valuable.go

packagemainimport"fmt"typestockPositionstruct{    tickerstringsharePricefloat32countfloat32}/*method to determine the value of a stock position*/func(sstockPosition)getValue()float32{returns.sharePrice* s.count}typecarstruct{makestringmodelstringpricefloat32}/*method to determine the value of a car*/func(ccar)getValue()float32{returnc.price}/*contract that defines different things that have value*/typevaluableinterface{getValue()float32}funcshowValue(assetvaluable) {    fmt.Printf("Value of the asset is%f\n", asset.getValue())}funcmain() {varovaluable = stockPosition{"GOOG",577.20,4}showValue(o)    o = car{"BMW","M3",66500}showValue(o)}

输出:

Value of the asset is 2308.800049

Value of the asset is 66500.000000

一个标准库的例子

io包里有一个接口类型Reader:

typeReaderinterface{Read(p []byte) (nint, errerror)}

定义变量r:var r io.Reader

那么就可以写如下的代码:

varrio.Readerr = os.Stdin//see 12.1r = bufio.NewReader(r)    r =new(bytes.Buffer)    f,_:=os.Open("test.txt")    r = bufio.NewReader(f)

上面r右边的类型都实现了Read()方法,并且有相同的方法签名,r的静态类型是io.Reader。

备注

有的时候,也会以一种稍微不同的方式来使用接口这个词:从某个类型的角度来看,它的接口指的是:它的所有导出方法,只不过没有显式地为这些导出方法额外定一个接口而已。

练习 11.1simple_interface.go:

定义一个接口Simpler,它有一个Get()方法和一个Set(),Get()返回一个整型值,Set()有一个整型参数。创建一个结构体类型Simple实现这个接口。

接着定一个函数,它有一个Simpler类型的参数,调用参数的Get()和Set()方法。在main函数里调用这个函数,看看它是否可以正确运行。

练习 11.2interfaces_poly2.go:

a) 扩展 interfaces_poly.go 中的例子,添加一个Circle类型

b) 使用一个抽象类型Shape(没有字段) 实现同样的功能,它实现接口Shaper,然后在其他类型里内嵌此类型。扩展 10.6.5 中的例子来说明覆写。

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

推荐阅读更多精彩内容

  • fmt格式化字符串 格式:%[旗标][宽度][.精度][arg索引]动词旗标有以下几种:+: 对于数值类型总是输出...
    皮皮v阅读 1,090评论 0 3
  • 出处---Go编程语言 欢迎来到 Go 编程语言指南。本指南涵盖了该语言的大部分重要特性 Go 语言的交互式简介,...
    Tuberose阅读 18,411评论 1 46
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,834评论 25 707
  • 第一次知道反射的时候还是许多年前在学校里玩 C# 的时候。那时总是弄不清楚这个复杂的玩意能有什么实际用途……然后发...
    勿以浮沙筑高台阅读 1,125评论 0 9
  • 要实现远大理想,成为一名伟大的政治家,军事家,诗人,政论家,首当其冲的要务是先要将自身塑造成一位思想家以及哲学家。...
    中庸阳阅读 271评论 0 0