在现实生活中,某些类具有两个或多个维度的变化,如图形既可按形状分,又可按颜色分。如何设计类似于 Photoshop 这样的软件,能画不同形状和不同颜色的图形呢?如果用继承方式,m 种形状和 n 种颜色的图形就有 m×n 种,不但对应的子类很多,而且扩展困难。
当然,这样的例子还有很多,如不同颜色和字体的文字、不同品牌和功率的汽车、不同类型的包和颜色。如果用桥接模式就能很好地解决这些问题。
桥接模式的定义与特点
桥接(Bridge
)模式的定义如下:将抽象与实现分离,使它们可以独立变化。它是用组合关系代替继承关系来实现,从而降低了抽象和实现这两个可变维度的耦合度。
桥接(Bridge)模式的优点
是:
1.由于抽象与实现分离,所以扩展能力强;
2.其实现细节对客户透明。
缺点
:
由于聚合关系建立在抽象层,要求开发者针对抽象化进行设计与编程,这增加了系统的理解与设计难度。
桥接模式的结构
可以将抽象化部分与实现化部分分开,取消二者的继承关系,改用组合关系。
桥接(Bridge)模式包含以下主要角色:
抽象化(Abstraction)角色:定义抽象类,并包含一个对实现化对象的引用。
扩展抽象化(Refined Abstraction)角色:是抽象化角色的子类,实现父类中的业务方法,并通过组合关系调用实现化角色中的业务方法。
实现化(Implementor)角色:定义实现化角色的接口,供扩展抽象化角色调用。
具体实现化(Concrete Implementor)角色:给出实现化角色接口的具体实现。
桥接模式的实现
举例:买包。包有背包、钱包之分,又有颜色之分:红色、黄色。
如果用继承的方式:
父类是“包”,子类有:“红色背包”、“红色钱包”、“黄色背包”、“黄色钱包”四个子类了。耦合度相当高。
使用桥接模式:
可以将“颜色”提取出来为“实现化角色”,将它与抽象化角色——“包”,进行一一组合就可以了。“颜色”+“包”,如:“红色”+“背包”=“红色背包”
/**
* 实现化角色:颜色
*/
public interface Color {
//获取颜色的方法
String getColor();
}
/**
* 具体实现化角色:红色
*/
public class Red implements Color {
public String getColor() {
return "红色";
}
}
/**
* 具体实现化角色:黄色
*/
public class Yellow implements Color {
public String getColor() {
return "黄色";
}
}
/**
* 抽象化角色:包
*/
public abstract class Bag {
//包含一个对实现化对象的引用
protected Color color;
protected Bag(Color color){
this.color=color;
}
//获取包的方法
public abstract String getBag();
}
/**
* 扩展抽象化角色:背包
* 实现父类中的业务方法,并通过组合关系调用实现化角色中的业务方法
*/
public class Backpack extends Bag{
public Backpack(Color color){
super(color);
}
public String getBag() {
return color.getColor()+"背包";//组合
}
}
/**
* 扩展抽象化角色:钱包
*/
public class Wallet extends Bag {
protected Wallet(Color color) {
super(color);
}
public String getBag() {
return color.getColor()+"钱包";
}
}
/**
* 测试类
*/
public class Test {
public static void main(String[] args) {
Backpack backpack1 = new Backpack(new Red());
System.out.println("买了一个:"+backpack1.getBag());
Backpack backpack2 = new Backpack(new Yellow());
System.out.println("买了一个:"+backpack2.getBag());
System.out.println("买了一个:"+new Wallet(new Red()).getBag());
System.out.println("买了一个:"+new Wallet(new Yellow()).getBag());
}
}
桥接模式的应用场景
桥接模式通常适用于以下场景:
1.当一个类存在两个独立变化的维度,且这两个维度都需要进行扩展时。
2.当一个系统不希望使用继承或因为多层次继承导致系统类的个数急剧增加时。
3.当一个系统需要在构件的抽象化角色和具体化角色之间增加更多的灵活性时。