问题由一个需求引起:
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
的。
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中要模拟普通的继承,除了使用嵌套之外,还需要在父类中“注册”子类的信息!