某些类具有两个或多个维度的变化,如图形既可按形状分,又可按颜色分。如果用继承方式,m 种形状和 n 种颜色的图形就有 m×n 种,不但对应的子类很多,而且扩展困难
如果能根据实际需要对形状和颜色进行组合。 则来进行设计系统中类的个数更少,且系统扩展更为方便。这种方式即桥接模式。桥接模式将继承关系转换为关联关系,从而降低了类与类之间的耦合,减少了代码编写量。
桥接模式的定义如下:将抽象与实现分离,使它们可以独立变化。它是用组合关系代替继承关系来实现,从而降低了抽象和实现这两个可变维度的耦合度。
模式的结构:桥接(Bridge)模式包含以下主要角色。
- 抽象化(Abstraction)角色:定义抽象类,并包含一个对实现化对象的引用。
- 扩展抽象化(Refined Abstraction)角色:是抽象化角色的子类,实现父类中的业务方法,并通过组合关系调用实现化角色中的业务方法。
- 实现化(Implementor)角色:定义实现化角色的接口,供扩展抽象化角色调用。
- 具体实现化(Concrete Implementor)角色:给出实现化角色接口的具体实现。
桥接模式的应用实例:用桥接(Bridge)模式模拟女士皮包的选购。
分析:女士皮包有很多种,可以按用途分、按皮质分、按品牌分、按颜色分等,本实例按用途分可选:钱包(Wallet)和挎包(HandBag),按颜色分可选:黄色(Yellow)和红色(Red)。
//实现化角色:颜色
interface Color {
String getColor();
}
//具体实现化角色:黄色
class Yellow implements Color {
public String getColor() {
return "yellow";
}
}
//具体实现化角色:红色
class Red implements Color {
public String getColor() {
return "red";
}
}
//抽象化角色:包
abstract class Bag {
protected Color color;
public void setColor(Color color) {
this.color = color;
}
public abstract String getName();
}
//扩展抽象化角色:挎包
class HandBag extends Bag {
public String getName() {
return color.getColor() + "HandBag";
}
}
//扩展抽象化角色:钱包
class WalletBag extends Bag {
public String getName() {
return color.getColor() + "Wallet";
}
}
测试:
public class BagTest {
public static void main(String[] args) {
//挎包搭配红色
Bag handBag = new HandBag();
handBag.setColor(new Red());
handBag.getName();
//钱包包搭配黄色
Bag walletBag = new WalletBag();
walletBag.setColor(new Yellow());
walletBag.getName();
}
}
桥接模式通常适用于以下场景:
- 当一个类存在两个及以上独立变化的维度,且这两个维度都需要进行扩展时。
- 当一个系统不希望使用继承或因为多层次继承导致系统类的个数急剧增加时。
- 当一个系统需要在构件的抽象化角色和具体化角色之间增加更多的灵活性时。
桥接(Bridge)模式的优点是:
- 抽象与实现分离,扩展能力强
- 符合开闭原则
- 符合合成复用原则
- 其实现细节对客户透明
缺点:由于聚合关系建立在抽象层,要求开发者针对抽象化进行设计与编程,能正确地识别出系统中两个独立变化的维度,这增加了系统的理解与设计难度。