概念
桥接模式(Bridge),将抽象部分与它的实现部分分离,使它们都可以独立的变化。
抽象部分是指上层使用的对象, 实现部分是指底层要调用的对象。 抽象部分和实现部分都是实现了相应的接口。
桥接模式类似于策略模式,区别在于:
策略模式:封装一系列算法使得算法可以互相替换, 共用一个方法来调用一系列算法。
桥接模式:使抽象部分和实现部分分离,可以独立变化, 每个抽象和实现都是一一对应。
这种模式涉及到一个作为桥接的接口,使得实体类的功能独立于接口实现类。这两种类型的类可被结构化改变而互不影响。
具体实现
- 定义一个操作接口, 接口有操作方法
- 多个具体操作对象实现操作接口
- 定义一个使用操作接口的接口
- 定义多个实现了使用操作接口的对象结构体, 结构体包含具体操作对象的接口。 在实现的接口中, 调用具体操作对象的接口。
模式的场景和优缺点
在有多种可能会变化的情况下,用继承会造成类爆炸问题,扩展起来不灵活
使用场景
实现系统可能有多个角度分类,每一种角度都可能变化。
优点
抽象和实现的分离、优秀的扩展能力、实现细节对客户透明
缺点
桥接模式的引入会增加系统的理解与设计难度,由于聚合关联关系建立在抽象层,要求开发者针对抽象进行设计与编程
代码实现
package main
import "fmt"
// ****************手机品牌接口****************
type PhoneBrand interface {
InstallSoft(soft PhoneSoft)
Run()
}
// ************************end***************************
// **********手机品牌N, 实现了手机品牌接口PhoneBrand***********
type PhoneBrandN struct {
soft PhoneSoft
}
func (h *PhoneBrandN) Run() {
h.soft.run()
}
func (h *PhoneBrandN) InstallSoft(soft PhoneSoft) {
h.soft = soft
}
// ************************end***************************
// **********手机品牌M, 实现了手机品牌接口PhoneBrand***********
type PhoneBrandM struct {
soft PhoneSoft
}
func (h *PhoneBrandM) Run() {
h.soft.run()
}
func (h *PhoneBrandM) InstallSoft(soft PhoneSoft) {
h.soft = soft
}
// ************************end***************************
// ****************手机软件接口****************
type PhoneSoft interface {
run()
}
// *******************end***********************
// ****************手机游戏软件, 实现了软件接口PhoneSoft****************
type PhoneGame struct {
}
func (h *PhoneGame) run() {
fmt.Println("运行手机游戏")
}
// *******************end***********************
// ****************手机通讯录软件, 实现了软件接口PhoneSoft****************
type PhoneAddressList struct {
}
func (h *PhoneAddressList) run() {
fmt.Println("运行手机通讯录")
}
// *******************end***********************
func main() {
//桥接模式
// 构建不同的手机品牌, 传递不同的软件, 运行
var ab PhoneBrand
// N品牌手机
ab = &PhoneBrandN{}
ab.InstallSoft(&PhoneGame{})
ab.Run()
ab.InstallSoft(&PhoneAddressList{})
ab.Run()
// M品牌手机
ab = &PhoneBrandM{}
ab.InstallSoft(&PhoneGame{})
ab.Run()
ab.InstallSoft(&PhoneAddressList{})
ab.Run()
}