Go语言学习(十三)面向对象编程-继承

1.匿名组合

Go语言也提供了继承,但是采用了组合的方式,所以我们将其称为匿名组合:

package main
import "fmt"

//定义基类
type Base struct {
    Name string
}
//基类相关的2个成员方法
func (base *Base) A() { 
    fmt.Println("Base method A  called...")
}
func (base *Base) B() { 
    fmt.Println("Base method B called...")
}

//定义子类
type Son struct {
    Base  //"继承"基类
}
//重写基类的B方法
func (son *Son) B() {
    son.Base.B() //调用基类的B方法
    fmt.Println("Son method B called...")
}

func main(){
    son := Son{Base{"mChenys"}}
    son.B() //调用子类的重写至基类的B方法
    son.A() //调用子类继承至基类的A方法
}

输出结果:

Base method B called...
Son method B called...
Base method A  called...

以上代码定义了一个Base类(实现了A()和B()两个成员方法),然后定义了一个
Son,该类从Base类“继承”并改写了B()方法(该方法实现时先调用了基类的B()方法).

在“派生类”Son没有改写“基类”Base的成员方法时,相应的方法就被“继承”,例如在
上面的例子中,调用son.A()和调用son.Base.A()效果一致。
与其他语言不同,Go语言很清晰地告诉你类的内存布局是怎样的。此外,在Go语言中你还
可以随心所欲地修改内存布局,如:

type Son struct {
    ... // 其他成员
    Base
}

这段代码从语义上来说,和上面给的例子并无不同,但内存布局发生了改变。“基类” Base
的数据放在了“派生类” Son的最后。

另外,在Go语言中,你还可以以指针方式从一个类型“派生”:
type Son struct {
*Base
…// 其他成员
}
这段Go代码仍然有“派生”的效果,只是Son创建实例的时候,需要外部提供一个Base类
实例的指针.
如下所示,匿名组合了一个log.Logger指针:

type Job struct {
    Command string
    *log.Logger
}

在合适的赋值后,我们在Job类型的所有成员方法中可以很舒适地借用所有log.Logger提
供的方法。比如如下的写法:
func (job *Job)Start() {
job.Log(“starting now…”) //注意:Log函数的接收者仍然是log.Logger指针
… // 做一些事情
job.Log(“started.”)
}
对于Job的实现者来说,他甚至根本就不用意识到log.Logger类型的存在,这就是匿名组合的
魅力所在.在实际工作中,只有合理利用才能最大发挥这个功能的价值。

2.名字冲突问题

我们必须关注一下接口组合中的名字冲突问题,比如如下的组合:

package main
import(
    "fmt"
)
type X struct {
    Name string
}
type Y struct {
    X
    Name string //相同名字的属性名会覆盖父类的属性
}
func main(){
    y := Y{X{"XChenys"},"YChenys"}
    fmt.Println("y.Name = ",y.Name) //y.Name = YChenys
}

组合的类型和被组合的类型都包含一个Name成员,会不会有问题呢?答案是否定的。所有
的Y类型的Name成员的访问都只会访问到最外层的那个Name变量,X.Name变量相当于被覆盖了。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值