描述
桥接模式是将抽象部分与它的实现部分分离,使它们都可以独立地变化。桥接模式是对象的结构模式。又称为柄体(Handle and Body)模式或接口(Interface)模式。
简介
桥接的用意是“将抽象化(Abstraction)与实现化(Implementation)脱耦。这句话有三个关键词,也就是抽象化、实现化和脱耦。
- 抽象化:从众多的事物中抽取出共同的、本质性的特征,而舍弃其非本质的特征。例如佳能相机,索尼相机,尼康相机,徕卡相机等,它们共同的特性就是相机。
- 实现化:针对抽象化给出的具体实现。它和抽象化是一个互逆的过程,实现化是对抽象化事物的进一步具体化。
- 脱耦:脱耦就是将抽象化和实现化之间的耦合解脱开,或者说是将它们之间的强关联改换成弱关联,将两个角色之间的继承关系改为关联关系。
角色
- 抽象化(Abstraction)角色:抽象化给出的定义,并保存一个对实现化对象的引用。
- 修正抽象化(RefinedAbstraction)角色:扩展抽象化角色,改变和修正父类对抽象化的定义。
- 实现化(Implementor)角色:这个角色给出实现化角色的接口,但不给出具体的实现。必须指出的是,这个接口不一定和抽象化角色的接口定义相同,实际上,这两个接口可以非常不一样。实现化角色应当只给出底层操作,而抽象化角色应当只给出基于底层操作的更高一层的操作。
- 具体实现化(ConcreteImplementor)角色:这个角色给出实现化角色接口的具体实现。
优缺点
优点
- 分离抽象接口及其实现部分。提高了比继承更好的解决方案。
- 桥接模式提高了系统的可扩充性,在两个变化维度中任意扩展一个维度,都不需要修改原有系统。
缺点
- 桥接模式的引入会增加系统的理解与设计难度,由于聚合关联关系建立在抽象层,要求开发者针对抽象进行设计与编程。
- 接模式要求正确识别出系统中两个独立变化的维度,因此其使用范围具有一定的局限性。
使用场景
- 如果一个系统需要在构件的抽象化角色和具体化角色之间增加更多的灵活性,避免在两个层次之间建立静态的继承联系,通过桥接模式可以使它们在抽象层建立一个关联关系。
- 对于那些不希望使用继承或因为多层次继承导致系统类的个数急剧增加的系统,桥接模式尤为适用。
- 一个类存在两个独立变化的维度,且这两个维度都需要进行扩展。
示例
假如有这样一个业务场景:导出项目的需求。
从业务上看,导出格式又分成word格式、execl格式、pdf格式等多种,不同的格式类型,业务功能处理是不一样的;从导出的手段上看,又有直接下载、发送到邮件、导出到系统个人空间内等。
/**
* 实现化(Implementor)角色
*/
public interface ExportWay {
void export(String requirement);
}
/**
* 具体实现化(ConcreteImplementor)角色
*/
public class EmailExport implements ExportWay {
@Override
public void export(String requirement) {
System.out.println("通过邮件导出需求:" + requirement);
}
}
/**
* 具体实现化(ConcreteImplementor)角色
*/
public class DownloadExport implements ExportWay {
@Override
public void export(String requirement) {
System.out.println("通过直接下载导出需求:" + requirement);
}
}
/**
* 具体实现化(ConcreteImplementor)角色
*/
public class EmailExport implements ExportWay {
@Override
public void export(String requirement) {
System.out.println("通过邮件导出需求:" + requirement);
}
}
/**
* 抽象化(Abstraction)角色
*/
public abstract class ExportRequirement {
/**
* 持有一个实现部分的对象
*/
private ExportWay exportWay;
public ExportRequirement(ExportWay exportWay) {
this.exportWay = exportWay;
}
public void exportFormat(String requirement) {
this.exportWay.export(requirement);
}
}
/**
* 修正抽象化(RefinedAbstraction)角色
*/
public class PdfFormat extends ExportRequirement {
public PdfFormat(ExportWay exportWay) {
super(exportWay);
}
@Override
public void exportFormat(String requirement) {
System.out.println("导出PDF格式的需求");
super.exportFormat(requirement);
}
}
/**
* 修正抽象化(RefinedAbstraction)角色
*/
public class WordFormat extends ExportRequirement{
public WordFormat(ExportWay exportWay) {
super(exportWay);
}
@Override
public void exportFormat(String requirement) {
System.out.println("导出word格式的需求");
super.exportFormat(requirement);
}
}
/**
* 修正抽象化(RefinedAbstraction)角色
*/
public class ExcelFormat extends ExportRequirement {
public ExcelFormat(ExportWay exportWay) {
super(exportWay);
}
@Override
public void exportFormat(String requirement) {
System.out.println("导出excel格式的需求");
super.exportFormat(requirement);
}
}