Golang之繼承模擬


問題由一個需求引起:

web的controller,希望創建一個基類,然后在子類的controller中定義action方法,基類有一個run函數能根據字符串自動找到子類的action方法。

 

如何解決呢? -- 用繼承

示例分析繼承

首先這個需求是很普遍的,由於腦中有繼承概念,所以想當然地以為這個很容易實現:

package main

import(
	"reflect"
)

type A struct {
}


func (self A)Run() {
	c := reflect.ValueOf(self)
	method := c.MethodByName("Test")
	println(method.IsValid())
}

type B struct {
	A
}

func (self B)Test(s string){
	println("b")
}

func main() {
	b := new(B)
	b.Run()
}

B繼承A,B中調用Run方法,自然會調用到A的Run方法,然后我根據string“Test”,希望能找到B中(B是子類)的Test方法。

 

 

用繼承的觀點看沒錯,實際運行呢?method.IsValid() 返回false。很明顯,這里的Test方法是找不到的。

 

分析問題,首先這里“繼承”兩個詞就用錯了,在go中不應該提及“繼承”這個詞,我更選擇使用“嵌套”這個詞。B是嵌套了A,所以這里的b.Run()實際上是語法糖,調用的是b.A.Run()。這里Run的全部環境都在A中。所以是找不到A的Test的。

 

感謝@hongqirui和@海意,在它們幫忙下找到了解決方法:

package main

import(
	"reflect"
)

type A struct {
	Parent interface{}
}


func (self A)Run() {
	c := reflect.ValueOf(self.Parent)
	method := c.MethodByName("Test")
	println(method.IsValid())
}

type B struct {
	A
}

func (self B)Test(s string){
	println("b")
}

func (self B)Run(){
	self.A.Run()
}


func main() {
	b := new(B)
	b.A.Parent = b
	b.Run()

}

在父類中加一個interface{}記錄子類!!這樣問題就迎刃而解了!method.IsValid()返回了true。

 

結論

所以在golang中要模擬普通的繼承,除了使用嵌套之外,還需要在父類中“注冊”子類的信息

 

外友情幫忙推薦下@海意 所在公司的團隊自產自銷的web框架golanger。https://github.com/golangers


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM