一篇文章帶你玩轉go語言的接口
一.其他語言
其他語言中所提供的接口概念:接口主要作為不同組件之間的契約存在。對契約的實現是強制的(侵入式接口),你必須聲明你的確實現瞭該接口。為瞭實現一個接口,你需要從該接口繼承。
interface IFoo {
void Bar();
}
// Java文法 // …
class Foo implements IFoo {
}
// C++文法 // …
class Foo : public IFoo {
}
“侵入式”的主要表現在於實現類需要明確聲明自己實現瞭 某個接口。
二.go語言
go語言中接口與其他語言的接口也略有不同,是一種非侵入式接口,實現類的時候,隻需要關心自己應該提供哪些方法,不用再糾結接口需要拆得多細才 合理。接口由使用方按需定義,而不用事前規劃。一個類隻需要實現瞭接口要求的所有函數,我們就說這個類實現瞭該接口。
type Phone interface { call() } type Nokia struct { name string } // 接口的實現是隱式的 func (phone Nokia) call() { fmt.Println("我是 Nokia,是一臺電話") }
三.go接口實現多態
package main import ( "fmt" "strconv" ) // 定義一個接口 type Good interface { settleAccount() int orderInfo() string } type Phone struct { name string quantity int price int } func (phone Phone) settleAccount() int { return phone.quantity * phone.price } func (phone Phone) orderInfo() string{ return "您要購買" + strconv.Itoa(phone.quantity)+ "個" + phone.name + "計:" + strconv.Itoa(phone.settleAccount()) + "元" } type FreeGift struct { name string quantity int price int } func (gift FreeGift) settleAccount() int { return 0 } func (gift FreeGift) orderInfo() string{ return "您要購買" + strconv.Itoa(gift.quantity)+ "個" + gift.name + "計:" + strconv.Itoa(gift.settleAccount()) + "元" } func calculateAllPrice(goods []Good) int { var allPrice int for _,good := range goods{ fmt.Println(good.orderInfo()) allPrice += good.settleAccount() } return allPrice } func main() { iPhone := Phone{ name: "iPhone", quantity: 1, price: 8000, } earphones := FreeGift{ name: "耳機", quantity: 1, price: 200, } goods := []Good{iPhone, earphones} allPrice := calculateAllPrice(goods) fmt.Printf("該訂單總共需要支付 %d 元", allPrice) }
四.空接口的使用(重點)
4.1定義
空接口沒有定義任何方法口,也因此,我們可以說所有類型都至少實現瞭空接口。
每一個接口都包含兩個屬性,一個是值,一個是類型。
而對於空接口來說,這兩者都是 nil,可以使用 fmt 來驗證一下
package main import ( "fmt" ) func main() { var i interface{} fmt.Printf("type: %T, value: %v", i, i) }
/type: <nil>, value: <nil>
4.2空接口使用
第一,通常我們會直接使用 interface{} 作為類型聲明一個實例,而這個實例可以承載任意類型的值。
// 聲明一個空接口實例 var i interface{} // 存 int 沒有問題 i = 1 fmt.Println(i) // 存字符串也沒有問題 i = "hello" fmt.Println(i) // 存佈爾值也沒有問題 i = false fmt.Println(i)
第二,如果想讓你的函數可以接收任意類型的值 ,也可以使用空接口
第三,你也定義一個可以接收任意類型的 array、slice、map、strcut,例如這邊定義一個切片
func main() { any := make([]interface{}, 5) any[0] = 11 any[1] = "hello world" any[2] = []int{11, 22, 33, 44} for _, value := range any { fmt.Println(value) } }
4.3空接口幾個要註意的坑(我剛學時的錯誤)
坑1:空接口可以承載任意值,但不代表任意類型就可以承接空接口類型的值
// 聲明a變量, 類型int, 初始值為1 var a int = 1 // 聲明i變量, 類型為interface{}, 初始值為a, 此時i的值變為1 var i interface{} = a // 聲明b變量, 嘗試賦值i 報錯 var b int = i
坑2:當空接口承載數組和切片後,該對象無法再進行切片
sli := []int{2, 3, 5, 7, 11, 13} var i interface{} i = sli //報錯 g := i[1:3] fmt.Println(g)
坑3:當你使用空接口來接收任意類型的參數時,它的靜態類型是 interface{},但動態類型(是 int,string 還是其他類型)我們並不知道,因此需要使用類型斷言。
這裡還有一點要說明 空接口調用函數時的隱式轉換 func myfunc(i interface{}) { switch i.(type) { case int: fmt.Println("參數的類型是 int") case string: fmt.Println("參數的類型是 string") } } func main() { a := 10 b := "hello" myfunc(a) myfunc(b) 如果寫在外面 則報錯 /*switch a.(type) { case int: fmt.Println("參數的類型是 int") case string: fmt.Println("參數的類型是 string") } */ }
1和3是最容易犯問題,唉。。。
總結
本篇文章就到這裡瞭,希望能夠給你帶來幫助,也希望您能夠多多關註WalkonNet的更多內容!
推薦閱讀:
- 深入瞭解Go語言的基本語法與常用函數
- go類型轉換及與C的類型轉換方式
- 基於Go Int轉string幾種方式性能測試
- java實現超市管理系統
- Go語言strconv包實現字符串和數值類型的相互轉換