概述
如果說(shuō)goroutine和channel是Go并發(fā)的兩大基石,那么接口是Go語(yǔ)言編程中數(shù)據(jù)類型的關(guān)鍵。在Go語(yǔ)言的實(shí)際編程中,幾乎所有的數(shù)據(jù)結(jié)構(gòu)都圍繞接口展開,接口是Go語(yǔ)言中所有數(shù)據(jù)結(jié)構(gòu)的核心。
Go語(yǔ)言中的接口是一些方法的集合(method set),它指定了對(duì)象的行為:如果它(任何數(shù)據(jù)類型)可以做這些事情,那么它就可以在這里使用。
接口的定義和使用
比如
復(fù)制代碼 代碼如下:
type I interface{
Get() int
Put(int)
}
這段話就定義了一個(gè)接口,它包含兩個(gè)函數(shù)Get和Put
好了,我的一個(gè)接口實(shí)現(xiàn)了這個(gè)接口:
復(fù)制代碼 代碼如下:
type S struct {val int}
func (this *S) Get int {
return this.val
}
func (this *S)Put(v int) {
this.val = v
}
這個(gè)結(jié)構(gòu)S就是實(shí)現(xiàn)了接口I
Go中interface的寫法
下面看幾個(gè)interface的例子:
復(fù)制代碼 代碼如下:
func SomeFunction(w interface{Write(string)}){
w.Write("pizza")
}
這個(gè)例子中,直接將interface定義在參數(shù)中,很特別…
復(fù)制代碼 代碼如下:
func weirdFunc( i int ) interface{} {
if i == 0 {
return "zero"
}
return i;
}
接口賦值
我們可以將一個(gè)實(shí)現(xiàn)接口的對(duì)象實(shí)例賦值給接口,也可以將另外一個(gè)接口賦值給接口。
(1)通過(guò)對(duì)象實(shí)例賦值
將一個(gè)對(duì)象實(shí)例賦值給一個(gè)接口之前,要保證該對(duì)象實(shí)現(xiàn)了接口的所有方法。考慮如下示例:
復(fù)制代碼 代碼如下:
type Integer int
func (a Integer) Less(b Integer) bool {
return a b
}
func (a *Integer) Add(b Integer) {
*a += b
}
type LessAdder interface {
Less(b Integer) bool
Add(b Integer)
}
var a Integer = 1
var b1 LessAdder = a //OK
var b2 LessAdder = a //not OK
b2的賦值會(huì)報(bào)編譯錯(cuò)誤,為什么呢?還記得類型方法>一章中討論的Go語(yǔ)言規(guī)范的規(guī)定嗎?
The method set of any other named type T consists of all methods with receiver type T. The method set of the corresponding pointer type T is the set of all methods with receiver T or T (that is, it also contains the method set of T).
也就是說(shuō)*Integer實(shí)現(xiàn)了接口LessAdder的所有方法,而Integer只實(shí)現(xiàn)了Less方法,所以不能賦值。
(2)通過(guò)接口賦值
復(fù)制代碼 代碼如下:
var r io.Reader = new(os.File)
var rw io.ReadWriter = r //not ok
var rw2 io.ReadWriter = new(os.File)
var r2 io.Reader = rw2 //ok
因?yàn)閞沒(méi)有Write方法,所以不能賦值給rw。
接口嵌套
我們來(lái)看看io package中的另外一個(gè)接口:
復(fù)制代碼 代碼如下:
// ReadWriter is the interface that groups the basic Read and Write methods.
type ReadWriter interface {
Reader
Writer
}
該接口嵌套了io.Reader和io.Writer兩個(gè)接口,實(shí)際上,它等同于下面的寫法:
復(fù)制代碼 代碼如下:
type ReadWriter interface {
Read(p []byte) (n int, err error)
Write(p []byte) (n int, err error)
}
注意,Go語(yǔ)言中的接口不能遞歸嵌套,
復(fù)制代碼 代碼如下:
// illegal: Bad cannot embed itself
type Bad interface {
Bad
}
// illegal: Bad1 cannot embed itself using Bad2
type Bad1 interface {
Bad2
}
type Bad2 interface {
Bad1
}
空接口(empty interface)
空接口比較特殊,它不包含任何方法:
復(fù)制代碼 代碼如下:
interface{}
在Go語(yǔ)言中,所有其它數(shù)據(jù)類型都實(shí)現(xiàn)了空接口。
復(fù)制代碼 代碼如下:
var v1 interface{} = 1
var v2 interface{} = "abc"
var v3 interface{} = struct{ X int }{1}
如果函數(shù)打算接收任何數(shù)據(jù)類型,則可以將參考聲明為interface{}。最典型的例子就是標(biāo)準(zhǔn)庫(kù)fmt包中的Print和Fprint系列的函數(shù):
復(fù)制代碼 代碼如下:
func Fprint(w io.Writer, a ...interface{}) (n int, err error)
func Fprintf(w io.Writer, format string, a ...interface{})
func Fprintln(w io.Writer, a ...interface{})
func Print(a ...interface{}) (n int, err error)
func Printf(format string, a ...interface{})
func Println(a ...interface{}) (n int, err error)
注意,[]T不能直接賦值給[]interface{}
復(fù)制代碼 代碼如下:
t := []int{1, 2, 3, 4}
var s []interface{} = t
編譯時(shí)會(huì)輸出下面的錯(cuò)誤:
cannot use t (type []int) as type []interface {} in assignment
我們必須通過(guò)下面這種方式:
復(fù)制代碼 代碼如下:
t := []int{1, 2, 3, 4}
s := make([]interface{}, len(t))
for i, v := range t {
s[i] = v
}
您可能感興趣的文章:- Golang 使用接口實(shí)現(xiàn)泛型的方法示例
- golang分層測(cè)試之http接口測(cè)試入門教程
- golang基礎(chǔ)之Interface接口的使用
- golang中interface接口的深度解析
- golang中接口對(duì)象的轉(zhuǎn)型兩種方式