废话:
我觉得要理解一种设计模式,除了掌握怎么实现这种设计模式之外,更重要的是这种设计模式的应用场景。但还多时候应用场景很难用一些话概括起来,就像桥接模式中说的功能层次跟实现层次需要分离的时候,比较难理解。所有我更倾向于“如果不这么做,会怎样”的分析,从而来理解“这么做”的好处。
解决的问题:
桥接模式还是比较有用的一种模式,解决的问题是将系统的功能层次跟实现方法层次分离。这里要先理解功能层次跟实现方法层次。
反面例子:
先看一个没有将这两种层次分离的系统例子(以书中原生例子为扩展)。
书中原生例子:“显示一些东西”,显示的东西可以是字符串,也可以是其他类型的,而显示的方法可以是单行显示,也可以是多行显示。
功能层次:显示字符串还是显示其他类型
实现方法层次:单行还是多行
正面例子:
假如没有使用桥接模式,则可能是如下的设计方式:
先设计一个抽象类Display类,功能上定义displayString方法,displayInt方法,实现上定义SingleDisplay和MultiDisplay方法,然后设计各种Display的子类,去组合这四个方法实现不同的需求,子类数量是 功能 X 方法(个)。假如再增加一个displayBoolean的方法,或者再增加一个TwoLineDisplay方法,则需要修改父类和全部子类。
接下来使用桥接模式:
功能层次上,设计一个Display父类,它有最基本的print方法来显示单行,实现一个子类CountDisplay增加一个multiDisplay函数来实现多行显示。
实现方法层次上,设计一个DisplayImpl父类,定义单行显示的方法rawPrint方法,实现一个具体的StringDisplayImpl(还可以增加IntDisplayImpl等等)
然后功能层次的父类Display持有实现方法层次的父类DisplayImpl的实例。
如果这样设计,那么当需要再增加双行显示的功能,或者再增加Boolean类型的输出方法时,只需要再两边分别扩展,而不需要修改现有的类。
public class Display {
private DisplayImpl impl;
public Display(DisplayImpl impl) {
this.impl = impl;
}
public void open() {
impl.rawOpen();
}
public void print() {
impl.rawPrint();
}
public void close() {
impl.rawClose();
}
public final void display() {
open();
print();
close();
}
}
public class CountDisplay extends Display {
public CountDisplay(DisplayImpl impl) {
super(impl);
}
public void multiDisplay(int times) { // 循环显示times次
open();
for (int i = 0; i < times; i++) {
print();
}
close();
}
}
public abstract class DisplayImpl {
public abstract void rawOpen();
public abstract void rawPrint();
public abstract void rawClose();
}
public class StringDisplayImpl extends DisplayImpl {
private String string; // 要显示的字符串
private int width; // 以字节单位计算出的字符串的宽度
public StringDisplayImpl(String string) { // 构造函数接收要显示的字符串string
this.string = string; // 将它保存在字段中
this.width = string.getBytes().length; // 把字符串的宽度也保存在字段中,以供使用。
}
public void rawOpen() {
printLine();
}
public void rawPrint() {
System.out.println("|" + string + "|"); // 前后加上"|"并显示
}
public void rawClose() {
printLine();
}
private void printLine() {
System.out.print("+"); // 显示用来表示方框的角的"+"
for (int i = 0; i < width; i++) { // 显示width个"-"
System.out.print("-"); // 将其用作方框的边框
}
System.out.println("+"); // 显示用来表示方框的角的"+"
}
}