Appearance
桥接模式
概述
现在有一个需求,需要创建不同的图形,并且每个图形都有可能会有不同的颜色,我们可以利用继承的方式来设计类的关系
可以发现有很多的类,加入需要再增加一个形状或者再增加一种颜色,就需要创建更多的类
试想,在一个有多种可能会变化的维度的系统中,用继承方式会造成类的爆炸,扩展起来不理你干活。每次在一个维度上新增一个具体实现都要增加多个子类。为了更加灵活的设计系统,我们此时可以考虑使用桥接模式
定义
将抽象与实现进行分离,使他们可以独立变化。它是用租户关系代替继承关系来实现,从而降低了抽象和实现这两个可变维度的耦合度
结构
桥接(Bridge)模式包含一下主要角色:
- 抽象化(Abstraction)角色:定义抽象类,并包含一个对实现化对象引用
- 扩展抽象化(Refined Abstraction)角色:是抽象化角色的子类,实现父类中业务方法,并通过组合关系调用实现化角色中的业务方法
- 实现化(Implementor)角色:定义实现化角色的接口,供扩展抽象化角色调用
- 具体实现化(Concrete Implementor)角色:给出实例化角色接口的具体实现
案例
视频播放器
需要开发一个跨平台视频播放器,可以再不同操作系统平台(如:Windows、Mac、Linux等)上播放多种格式的视频文件,常见的视频格式包括rmvb、mp4、avi、wmv等等。该播放器包含了两个维护,适合使用桥接模式。其UML类图如下:
代码
抽象化角色(OperatingSystem)
java
package com.huangjiliang.design.heima.structure.bridge;
/**
* 系统【抽象化角色】
*/
public abstract class OperatingSystem {
protected VideoFile videoFile;
public OperatingSystem(VideoFile videoFile) {
this.videoFile = videoFile;
}
public abstract void play(String fileName);
}
扩展抽象化角色(Windows、Mac)
java
package com.huangjiliang.design.heima.structure.bridge;
/**
* Mac系统【扩展抽象化】
*/
public class Mac extends OperatingSystem {
public Mac(VideoFile videoFile) {
super(videoFile);
}
@Override
public void play(String fileName) {
videoFile.decode(fileName);
}
}
package com.huangjiliang.design.heima.structure.bridge;
/**
* Windows系统【扩展抽象化】
*/
public class Windows extends OperatingSystem {
public Windows(VideoFile videoFile) {
super(videoFile);
}
@Override
public void play(String fileName) {
videoFile.decode(fileName);
}
}
实现化角色(VideoFile)
java
package com.huangjiliang.design.heima.structure.bridge;
/**
* 视频文件【实现化角色】
*/
public interface VideoFile {
void decode(String fileName);
}
具体实现化角色(AVIFile、RMVBFile)
java
package com.huangjiliang.design.heima.structure.bridge;
/**
* 具体实现化角色
*/
public class AVIFile implements VideoFile{
@Override
public void decode(String fileName) {
System.out.println("AVI视频文件:" + fileName);
}
}
package com.huangjiliang.design.heima.structure.bridge;
public class RMVBFile implements VideoFile{
@Override
public void decode(String fileName) {
System.out.println("RMVB视频文件:" + fileName);
}
}
客户端(Client)
java
package com.huangjiliang.design.heima.structure.bridge;
/**
* 演示桥接模式客户端
*/
public class Client {
public static void main(String[] args) {
OperatingSystem system = new Windows(new AVIFile());
system.play("湄公河行动");
OperatingSystem os = new Mac(new RMVBFile());
os.play("八百");
}
}
优点
桥接模式提高了系统的可扩展性,在两个变化维度中任意扩展一个维度,都不需要修改原有系统
如:如果现在还有一种视频文件类型MP4,我们只需要再定义一个类实现VideoFile接口即可,其他类不需要发生变化;或者增加一个操作系统Linux,我们只需要增加Linux类继承OperatingSystem,其他类不需要发生变化
实现细节对客户端透明
使用场景
- 当一个类存在两个独立变化的维度,且这两个维度都需要进行扩展时
- 当一个系统不希望使用继承或者因为多层次继承导致系统类的个数急剧增加时
- 当一个系统需要在构建的抽象化角色和具体化角色之间增加更多的灵活性时、避免在两个层次之间建立静态的继承联系,通过桥接模式可以使它们在抽象层建立一个关联关系