设计模式13---桥接模式(Bridge Pattern)
桥接模式将抽象与具体实现分离,使得抽象与具体实现可以各自改变互不影响。桥接模式属于设计模式中的结构模式。
桥梁模式涉及的角色
抽象(Abstraction)角色:抽象定义,引用对接口对象的引用。
重新抽象(RefinedAbstraction)角色:重新抽象化角色,对父类中抽象的重新定义。
待实现接口(Implementor)角色:定义待实现的接口。
具体实现(ConcreteImplementor)角色:这个给出实现化角色接口的具体实现。
实例
上面的角色描述对应下图实例可以很容易的理解
这样设计的好处更好的理解面向对象编程,而且可以通过组合聚合的形式来完成扩展。
应用场景
会有多种类型的对象存在并且不断扩展(类似List的及其子类)
解耦两个模块或者系统(类似于DriverManager)
总结
这个模式实际开发中不是那么经常使用,不过对于理解面向对象编程是一个很好的模式,可以利用这种模式来完成解耦,并且是整个程序具有良好的扩展性。
桥梁模式是对象的结构模式。又称为柄体(Handle and Body)模式或接口(Interface)模式。桥梁模式的用意是“将抽象化(Abstraction)与实现化(Implementation)脱耦,使得二者独立的变化”。
桥梁模式的用意
桥梁模式虽然不是一个使用频率很高的模式,但是熟悉这个模式对于理解面向对象的设计原则,包括“开-闭”原则以及组合/聚合复用原则都很有帮助,理解好这两个原则,有助于形成正确的设计思想和培养良好的设计风格。
桥梁模式的用意是“将抽象化(Abstraction)与实现化(Implementation)脱耦,使得二者独立的变化”。这句话很短,但是第一次读到这句话的人很可能都会思考良久而不得其解。
这句话有三个关键词,也就是抽象化、实现化和脱耦。理解这三个词所代表的概念是理解桥梁模式用意的关键。
抽象化
从众多的事物中抽取出共同的、本质性的特征,而舍弃其非本质的特征,就是抽象化。例如:苹果、香蕉、梨、桃子等。它们共同的特征就是水果。得出水果概念的过程,就是一个抽象化的过程。要抽象,就必须进行比较,没有比较就无法找到在本质上共同的部分。共同特征是指那些能把一类事物与他类事物分开来的特征,这些具有区分作用的特征�又称为本质特征。因此抽取事物的共同特征就是抽取事物的本质特征,舍弃非本质的特征。所以抽象化的过程也是一个裁剪的过程。在抽象时,同与不同,决定于从什么角度上来抽象。抽象的角度决定于分析问题的目的。
通常情况下,一组对象如果具有相同的特征,那么它们就可以通过一个共同的类来描述。如果一些类具有相同的特征,往往可以通过一个共同的抽象类来描述。
实现化
抽象化给出的具体实现,就是实现化。
一个类的实例就是这个类的实例化,一个具体子类就是它的抽象超类的实例化。
脱耦
所谓耦合,就是两个实体的行为的某种强关联。而将它们的强关联去掉,就是耦合的脱耦。在这里,脱耦是指将抽象化和实例化之间的耦合解脱开,或者是将它们之间的强关联改成弱关联。
所谓强关联,就是在编译时期已经确定的,无法在运行时期动态改变的关联;所谓弱关联,就是可以动态确定并且可以在运行时期动态改变的关联。显然,在java语言中,继承关系是强关联,而聚合关系是弱关联。
将两个角色之间的继承关系改为聚合关系,就是将它们之间的强关联改换成为弱关联,因此,桥梁模式中的所谓脱耦,就是指在一个软件系统中的抽象化和实现化之间使用聚合关系,而不是继承关系,从而使两者可以相对对立的变化。
这就是桥梁模式的用意。
桥梁模式的结构
下图所示就是一个实现了桥梁模式的示意性系统的结构图:
可以看出,这个系统含有两个等级结构
- 由抽象化角色和修正抽象化角色组成的抽象化等级结构。
- 由实现化角色和两个具体实现化角色所组成的实现化等级结构。
桥梁模式涉及的角色有:
- 抽象化角色(Abstraction):抽象化给出的定义,并保存一个对实现化对象的引用。
- 修正抽象化角色(RefineAbstraction):拓展抽象化角色,改变和修正父类对抽象化的定义。
- 实现化角色(Implementor):这个角色给出实现化角色的接口,但是不给出具体的实现。必须指出的是,这个接口不一定和抽象化角色的定义相同,实际上,这两个接口可以非常不一样。实现化角色应当只给出底层操作,而抽象化角色应当只给出基于底层操作的更高一层的操作。
- 具体实现化角色(ConcreteImplementor):这个角色给出实现化角色接口的具体实现。
抽象化角色就像是一个水杯的手柄,而实现化角色和具体实现化角色就像是水杯的被神。手柄控制杯身,这就是此模式别名“柄体”的来源。
对象是行为的封装,而行为是由方法实现的。在这个示意性系统里,抽象化等级结构中的类封装了operation()
方法;而实现化等级结构中的类封装的是operationImpl()
方法。当然,在实际的系统中往往会有多于一个的方法。
抽象化等级结构中的方法通过向对应的实现化对象的委派实现自己的功能,这意味着抽象化角色可以通过向不同的实现化对象委派,来达到动态的转换自己的功能的目的。
示例代码
实现化角色
public abstract class Implementor {
/**
* 抽象方法,实现抽象部分需要的某些功能
*/
public abstract void operationImpl();
}
具体实现化角色
public class ConcreteImplementorA extends Implementor {
@Override
public void operationImpl() {
//具体操作
}
}
public class ConcreteImplementorB extends Implementor {
@Override
public void operationImpl() {
//具体操作
}
}
抽象化角色类,它声明了一个方法operation()
,并给出了它的实现。这个实现是通过向实现化对象的委派(也就是调用实现化对象的operationImpl()
方法)实现的。
public abstract class Abstraction {
protected Implementor implementor;
public Abstraction(Implementor implementor) {
this.implementor = implementor;
}
/**
* 示例方法
*/
public void operation() {
implementor.operationImpl();
}
}
修正抽象化角色
public class RefinedAbstraction extends Abstraction {
public RefinedAbstraction(Implementor implementor) {
super(implementor);
}
/**
* 其他的操作方法
*/
public void otherOperation() {
}
}
一般而言,实现化角色中的每个方法都应当有一个抽象化角色中的某一个方法与之对应,但是反过来则不一定。换而言之,抽象化角色的接口比实现化角色的接口宽。抽象化角色除了提供与实现化角色相关的方法之外,还有可能提供其他的方法;而实现化角色则往往仅为实现抽象化角色的相关行为而存在。
使用场景
考虑这样一个实际的业务功能:发送提示消息。基本上所有带业务流程处理的系统都会有这样的功能,比如OA上有尚未处理完毕的文件,需要发送一条消息提示他。
从业务上看,消息又分为普通消息、加急消息和特急消息多种,不同的消息类型,业务功能处理是不一样的,比如加急消息是在消息上添加加急,而特急消息除了添加特急外,还会做一条催促的记录,多久不完成会继续催促;从发送消息的手段上看,又有系统内短消息、手机短消息、邮件等。
不使用模式的解决方案
实现发送普通消息
先考虑实现一个简单点的版本,比如,消息只是实现发送普通消息,发送的方式只实现系统内短消息和邮件。其他的功能,等这个版本完成后,再继续添加。
示例代码
消息的统一接口
public interface Message {
/**
* 发送消息
* @param message 发送消息的内容
* @param toUser 接收人
*/
void send (String message, String toUser);
}
系统内短消息示例类
public class CommonMessageSMS implements Message {
@Override
public void send(String message, String toUser) {
System.out.println(String.format("使用系统内部短消息的方法,发送消息 %s 给 %s", message, toUser));
}
}
邮件短消息示例类
public class CommonMessgeEmail implements Message {
@Override
public void send(String message, String toUser) {
System.out.println(String.format("使用邮件短消息的方法,发送消息 %s 给 %s", message, toUser));
}
}
实现发送加急消息
发送加急消息同样有两种方式,系统内短消息和邮件方式。但是加急消息的实现不同于普通消息,加急消息会自动在消息上添加加急,然后再发送消息;另外加急消息会提供监控的方法,让客户端可以随时通过这个方法来了解对于加急消息的处理进度。比如,相应的人员是否接收到这个消息,相应的处理工作是否已经展开。因此加急消息需要拓展出一个新的接口,除了基本的发送消息的功能,还需要添加监控功能。
示例代码
加急消息的接口
public interface UrgencyMessage extends Message {
/**
* 监控指定消息的处理过程
* @param messageId 被监控的消息编号
* @return 监控到的消息的处理状态
*/
Object watch(String messageId);
}
系统内加急短消息示例类
public class UrgencyMessageSMS implements UrgencyMessage {
@Override
public void send(String message, String toUser) {
message = "加急:" + message;
System.out.println(String.format("使用系统内部短消息的方法,发送消息 %s 给 %s", message, toUser));
}
@Override
public Object watch(String messageId) {
//根据消息编码获取消息的状态,组成监控的数据对象,然后返回
return null;
}
}
邮件加急短消息示例类
public class UrgencyMessageEmail implements UrgencyMessage {
@Override
public void send(String message, String toUser) {
message = "加急:" + message;
System.out.println(String.format("使用邮件短消息的方法,发送消息 %s 给 %s", message, toUser));
}
@Override
public Object watch(String messageId) {
//根据消息编码获取消息的状态,组成监控的数据对象,然后返回
return null;
}
}
实现发送特急消息
特急消息不需要查看处理进程,只有没有完成,就直接催促,也就是说,对于特急消息,在普通消息的处理基础上,需要添加催促的功能。
观察上面的系统结构图,会发现一个很明显的问题,那就是通过这种继承的方式来拓展消息处理,会非常不方便。实现加急消息处理的时候,必须实现系统内短消息和邮件两种处理方式,因为业务处理可能不同,在实现特急消息处理的时候,又必须实现系统内短消息和邮件两种处理方式。这意味着,以后每次拓展一次消息处理,都必须要实现这两种处理方式,这还不算完,如果要添加新的处理方式呢?
添加发送手机消息的处理方式
如果要添加一种新的发送消息的方式,是需要在每一种抽象的具体实现中,都添加发送手机消息的处理的。也就是说,发送普通消息、加急消息、特急消息的处理,都可以通过手机来发送。
采用通过继承来扩展的实现方式,有个明显的缺点,扩展消息的种类不太容易。不同种类的消息具有不同的业务,也就是有不同的实现。在这种情况下,每一种类的消息,需要实现所有不同的消息发送方式。更可怕的是,如果要新加入一种消息的发送方式,那么会要求所有的消息种类都有加入这种新的发送方式的实现。
那么究竟该如何才能既实现功能,又可以灵活的拓展呢?
使用桥梁模式来解决问题
根据业务的功能要求,业务的变化具有两个维度,一个维度是抽象的消息,包括普通消息、加急消息和特急消息,这几个抽象的消息本身就具有一定的关系,加急消息和特急消息会拓展普通消息;另一个维度是在具体的消息发送方式上,包括系统内短消息、邮件短消息和手机短消息,这几个方式是平等的,可被切换的方式。
![消息发送维度解析](http://chuantu.biz/t5/137/1500211991x3683574917.png)
示例代码
实现消息发送的�统一接口
public interface MessageImplementor {
/**
* 发送消息方法
* @param message 要发送消息的内容
* @param toUser 接收人
*/
void send(String message, String toUser);
}
抽象消息类
public abstract class AbstractMessage {
/**
* 持有一个实现部分的对象
*/
MessageImplementor implementor;
/**
* 构造方法,传入实现部分的对象
* @param implementor 实现部分的对象
*/
public AbstractMessage(MessageImplementor implementor) {
this.implementor = implementor;
}
/**
* 发送消息,委派给实现部分的方法
* @param message 要发送的消息
* @param toUser 接收人
*/
public void sendMessage(String message, String toUser) {
this.implementor.send(message, toUser);
}
}
普通消息类
public class CommonMessage extends AbstractMessage {
/**
* 构造方法,传入实现部分的对象
*
* @param implementor 实现部分的对象
*/
public CommonMessage(MessageImplementor implementor) {
super(implementor);
}
@Override
public void sendMessage(String message, String toUser) {
//对于普通消息,直接调用父类方法,发送消息即可
super.sendMessage(message, toUser);
}
}
加急消息类
public class UrgencyMessage extends AbstractMessage {
/**
* 构造方法,传入实现部分的对象
*
* @param implementor 实现部分的对象
*/
public UrgencyMessage(MessageImplementor implementor) {
super(implementor);
}
@Override
public void sendMessage(String message, String toUser) {
message = "加急:" + message;
super.sendMessage(message, toUser);
}
/**
* 扩展它自己的功能,监控某个消息的处理状态
* @param messageId 消息编码
* @return 监控到的消息的处理状态
*/
public Object watch(String messageId) {
//根据给出的消息编码查询消息的处理状态,组织成监控的处理状态,然后返回。
return null;
}
}
系统内短消息的实现类
public class MessageSMS implements MessageImplementor {
@Override
public void send(String message, String toUser) {
System.out.println(String.format("使用系统内部短消息的方法,发送消息 %s 给 %s", message, toUser));
}
}
邮件短消息的实现类
public class MessageEmail implements MessageImplementor {
@Override
public void send(String message, String toUser) {
System.out.println(String.format("使用邮件短消息的方法,发送消息 %s 给 %s", message, toUser));
}
}
客户端类
public class Client {
public static void main (String[] args) {
MessageImplementor implementor = new MessageSMS();
AbstractMessage abstractMessage = new CommonMessage(implementor);
abstractMessage.sendMessage("加班申请速批", "陈总");
implementor = new MessageEmail();
abstractMessage = new UrgencyMessage(implementor);
abstractMessage.sendMessage("加班申请速批", "陈总");
}
}
通过上面的例子会发现,采用桥梁模式来实现,抽象部分和实现部分分离开了,可以相互独立的变化,而不会相互影响。因此在抽象部分增加新的消息处理(特急消息),对发送消息的实现部分是没有影响的;反过来增加发送消息的方式(手机短消息),对消息处理部分也是没有影响的。
桥梁模式的优点
- 分离抽象部分和实现部分
桥梁模式分离了抽象部分和实现部分,从而极大的提高了系统的灵活性。让抽象部分和实现部分独立出来,分别定义接口,这有助于对系统进行分层,从而产生更好的结构化的系统。 -
更好的扩展性
桥梁模式使得抽象部分和实现部分可以分别独立的扩展,而不会相互影响,从而大大提高了系统的可扩展性。
桥梁模式在Java中的使用
桥梁模式在Java应用中一个非常典型的例子就是JDBC驱动器。JDBC为所有的关系型数据库提供一个通用的界面。一个应用系统动态的选择一个合适的驱动器,然后通过驱动器向数据库引擎发出指令。这个过程就是将抽象角色的行为委派给实现角色的过程。
抽象角色可以针对任何数据库引擎发出查询指令,因为抽象角色并不直接�与数据库引擎打交道,JDBC驱动器负责这个底层的工作。由于JDBC驱动器的存在,应用系统可以不依赖于数据库引擎的细节而独立的演化;同时数据库引擎也可以独立�于应用系统的细节而独立的演化。两个独立的等级结构图如下图所示,左边是JDBC API的等级结构,右边是JDBC驱动器的等级结构,应用程序是建立在JDBC API的基础之上的。
应用系统作为一个等级结构,与JDBC驱动器这个等级结构是相对独立的,它们之间没有静态的强关联。应用系统通过委派与JDBC驱动器相互作用,这是一个桥梁模式的例子。
JDBC这种架构,把抽象部分和具体实现部分分离开来,从而使得抽象部分和具体实现部分都可以独立的扩展。对于应用程序而言,之遥选用不同的驱动,就可以让程序操作不同的数据库,而无需更改应用程序,从而实现在不同的数据库上移植;对于驱动程序而言,为数据库实现不同的驱动程序,并不会影响应用程序。