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