图解设计模式读书笔记(六)——Bridge(桥接)模式

废话:

我觉得要理解一种设计模式,除了掌握怎么实现这种设计模式之外,更重要的是这种设计模式的应用场景。但还多时候应用场景很难用一些话概括起来,就像桥接模式中说的功能层次跟实现层次需要分离的时候,比较难理解。所有我更倾向于“如果不这么做,会怎样”的分析,从而来理解“这么做”的好处。

解决的问题:

桥接模式还是比较有用的一种模式,解决的问题是将系统的功能层次实现方法层次分离。这里要先理解功能层次跟实现方法层次。

反面例子:

先看一个没有将这两种层次分离的系统例子(以书中原生例子为扩展)。

书中原生例子:“显示一些东西”,显示的东西可以是字符串,也可以是其他类型的,而显示的方法可以是单行显示,也可以是多行显示。

功能层次:显示字符串还是显示其他类型

实现方法层次:单行还是多行

正面例子:

假如没有使用桥接模式,则可能是如下的设计方式:

先设计一个抽象类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("+");                        // 显示用来表示方框的角的"+"
    }
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值