Go 分布式学习利器(13)-- Go语言的多态
生活随笔
收集整理的這篇文章主要介紹了
Go 分布式学习利器(13)-- Go语言的多态
小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.
文章目錄
- 1. 基本的多態(tài)實(shí)現(xiàn)
- 2. 空接口與斷言
- 3. Go接口的最佳實(shí)踐
1. 基本的多態(tài)實(shí)現(xiàn)
我們知道C++中實(shí)現(xiàn)多態(tài)是通過虛函數(shù)表 和 繼承來 實(shí)現(xiàn)的。
類似如下代碼:
class Programmar{
public:virtual void write_hello_world() = 0;
}class GoProgrammar: public Programmar{
public:void write_hello_world() {printf("Go write hello workld\n");}
}class CXXProgrammar: public Programmar{
public:void write_hello_world() {printf("C++ write hello workld\n");}
}class JavaProgrammar: public Programmar{
public:void write_hello_world() {printf("Java write hello workld\n");}
}void write_program(Programmar p) {p.write_hello_world();
}
通過將不同類型的Programmar子類對(duì)象傳入到write_program函數(shù)中即能夠調(diào)用對(duì)應(yīng)子類的實(shí)現(xiàn),達(dá)到使用父類的成員調(diào)用子類方法的目的,且調(diào)用過程是動(dòng)態(tài)綁定的,即在真正開始運(yùn)行的時(shí)候父類對(duì)象才能夠知道自己真正的類型是什么。
Go語言不支持繼承,那么Go語言是否也能夠?qū)崿F(xiàn)如上C++/Java類似的多態(tài)呢?
模擬以上代碼,畫出如下類圖,希望Go語言能夠?qū)崿F(xiàn)該類圖中的多態(tài)功能。
如下代碼:
package orientedimport ("fmt""testing"
)type Code string
type Programmar interface {WriteHelloWorld() Code
}type GoProgrammar struct {
}func (p *GoProgrammar) WriteHelloWorld() Code{return "fmt.Println(\"Go write hello world\")";
}type JavaProgrammar struct {
}func (p *JavaProgrammar) WriteHelloWorld() Code{return "fmt.Println(\"Java write hello world\")";
}func writeFirstProgram(p Programmar) {fmt.Printf("%T %v\n",p,p.WriteHelloWorld())
}func TestMulti(t *testing.T){// new 關(guān)鍵字創(chuàng)建的實(shí)例是指針類型,同時(shí)GoProgrammar是interface 結(jié)構(gòu),也是只能對(duì)應(yīng)指針類型的實(shí)例goPro := new(GoProgrammar)javaPro := new(JavaProgrammar)writeFirstProgram(goPro)writeFirstProgram(javaPro)
}
最后輸出如下:
=== RUN TestMulti
*oriented.GoProgrammar fmt.Println("Go write hello world")
*oriented.JavaProgrammar fmt.Println("Java write hello world")
--- PASS: TestMulti (0.00s)
這樣的多態(tài)實(shí)現(xiàn),顯然是OK的。
2. 空接口與斷言
- 空接口可以表示 任何類型。
p interface{}表示聲明了一個(gè)空接口變量 p - 通過斷言 可以將空接口轉(zhuǎn)為指定類型
類似:v,ok := p.(int),其中ok表示是否轉(zhuǎn)換成功,成功的話v中才有值
如下測(cè)試代碼:
package empty_interfaceimport ("fmt""testing"
)func Dosomething(p interface{}){if v,ok := p.(int); ok{fmt.Println("Integer :", v)return}if v,ok := p.(string); ok{fmt.Println("String :", v)return}fmt.Print("Unkown\n")
}// 重構(gòu)以上代碼,使用switch 關(guān)鍵字看起來會(huì)更加簡潔
func DosomthingSwitch(p interface{}) {switch v := p.(type) {case int:fmt.Println("Integer :", v)case string:fmt.Println("String :", v)default:fmt.Print("Unkown\n")}
}func TestInterface(t *testing.T) {Dosomething(10);Dosomething("test");DosomthingSwitch(111)
}
最后的輸出如下:
=== RUN TestInterface
Integer : 10
String : test
Integer : 111
--- PASS: TestInterface (0.00s)
3. Go接口的最佳實(shí)踐
為了讓Go代碼看起來更整潔,有以下幾個(gè)關(guān)于接口的編程規(guī)范/建議:
- 在Go語言中 傾向使用較小的接口定義,很多接口只包含一個(gè)方法。
type Reader interface{Read(p []byte) (n int, err error) } type Writer interface{Write(p []byte) (n int, err error) } - 較大的接口則可以由更多較小的接口組成
type FileOp interface {ReaderWriter } - 只依賴與必要功能的最小接口
其中func getData(read Reader) error {... }getData函數(shù)用到了Reader接口,則只需要傳入Reader參數(shù)即可,不需要將FileOp都傳入進(jìn)來,從而降低接口的可復(fù)用性。意思是說,如果傳入了FileOp,因?yàn)樗藘蓚€(gè)接口類型,那么兩個(gè)接口類型的函數(shù)都需要被實(shí)現(xiàn),但其實(shí)只用到Reader的函數(shù),這就很不友好。
總結(jié)
以上是生活随笔為你收集整理的Go 分布式学习利器(13)-- Go语言的多态的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Go 分布式学习利器(12)-- Go语
- 下一篇: Go 分布式学习利器(14)-- Go语