初步解讀Golang中的接口相關編寫方法

初步解讀Golang中的接口相關編寫方法


概述
如果說goroutine和channel是Go併發的兩大基石,那麼接口是Go語言編程中數據類型的關鍵。在Go語言的實際編程中,幾乎所有的數據結構都圍繞接口展開,接口是Go語言中所有數據結構的核心。
Go語言中的接口是一些方法的集合(method set),它指定了對象的行爲:如果它(任何數據類型)可以做這些事情,那麼它就可以在這裏使用。

接口的定義和使用

比如


複製代碼代碼如下:


type I interface{
    Get() int
    Put(int)
 
}




這段話就定義了一個接口,它包含兩個函數Get和Put


好了,我的一個接口實現了這個接口:


複製代碼代碼如下:


type S struct {val int}
func (this *S) Get int {
    return this.val
}
func (this *S)Put(v int) {
    this.val = v
 
}




這個結構S就是實現了接口I


Go中interface的寫法

下面看幾個interface的例子:


複製代碼代碼如下:



func SomeFunction(w interface{Write(string)}){
    w.Write("pizza")
 
}



這個例子中,直接將interface定義在參數中,很特別…



複製代碼代碼如下:


func weirdFunc( i int ) interface{} {
  if i ==  0 {
    return "zero"
  }
  return i;
}




接口賦值
我們可以將一個實現接口的對象實例賦值給接口,也可以將另外一個接口賦值給接口。


(1)通過對象實例賦值

將一個對象實例賦值給一個接口之前,要保證該對象實現了接口的所有方法。考慮如下示例:


複製代碼代碼如下:


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的賦值會報編譯錯誤,爲什麼呢?還記得<類型方法>一章中討論的Go語言規範的規定嗎?


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).
也就是說*Integer實現了接口LessAdder的所有方法,而Integer只實現了Less方法,所以不能賦值。

(2)通過接口賦值


複製代碼代碼如下:



        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



因爲r沒有Write方法,所以不能賦值給rw。


接口嵌套
我們來看看io package中的另外一個接口:


複製代碼代碼如下:



// ReadWriter is the interface that groups the basic Read and Write methods.
type ReadWriter interface {
 Reader
 Writer
}



該接口嵌套了io.Reader和io.Writer兩個接口,實際上,它等同於下面的寫法:



複製代碼代碼如下:



type ReadWriter interface {
Read(p []byte) (n int, err error) 
Write(p []byte) (n int, err error)
}



注意,Go語言中的接口不能遞歸嵌套,



複製代碼代碼如下:



// 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)
空接口比較特殊,它不包含任何方法:


複製代碼代碼如下:

interface{}


在Go語言中,所有其它數據類型都實現了空接口。



複製代碼代碼如下:



var v1 interface{} = 1
var v2 interface{} = "abc"
var v3 interface{} = struct{ X int }{1}



如果函數打算接收任何數據類型,則可以將參考聲明爲interface{}。最典型的例子就是標準庫fmt包中的Print和Fprint系列的函數:



複製代碼代碼如下:



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{}



複製代碼代碼如下:



        t := []int{1, 2, 3, 4}
        var s []interface{} = t



編譯時會輸出下面的錯誤:


cannot use t (type []int) as type []interface {} in assignment

我們必須通過下面這種方式:


複製代碼代碼如下:



t := []int{1, 2, 3, 4}
s := make([]interface{}, len(t))
for i, v := range t {
    s[i] = v
}


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章