程序员

桥接模式

2019-03-09  本文已影响13人  凯玲之恋

设计模式

一、引言

假如有三个品牌的手机vivo,oppo和小米,如果手机手机壳一体生产,会是这样的:

7017386-99137fe056cae28c.png

对应到相应的类中,将是1+3+6=10个有继承关系的类,如果这时再加一个华为手机,无疑是要多增加3个类,会带来类的急剧增长。

如果手机手机壳分开生产搭配(现实也的确是这样的),就是这样的:


7017386-9a0491b22387c445.png

对应到类设计中,只需要7个类,如果增加一类手机,只需增加一个类,增加一款手机壳,也只需增加一个类。

这种处理多维变化(手机和手机壳)的方式运用到软件设计中就是桥接模式。

二、什么是桥接模式

桥接模式是一种很实用的结构型设计模式,在软件开发时,如果某个类存在两个独立变化的维度,可以运用桥接模式将这两个维度分离出来,使两者可以独立扩展,让系统更加符合“单一职责原则”。

与多层继承方案不同,它将两个独立变化的维度设计为两个独立的继承等级结构,并且在抽象层建立一个抽象关联,该关联关系就像一条桥一样,将两个独立继承结构的类联接起来,故名桥接模式。

可以明显看出,桥接模式使用组合代替了继承,将类之间的静态继承关系转换为动态的对象组合关系,使用组合而不用继承,会使系统更加灵活,并易于扩展,同时有效控制了系统中类的个数

桥接定义如下:

桥接模式(Bridge Pattern):将抽象部分与它的实现部分分离,使它们都可以独立地变化。它是一种对象结构型模式,又称为柄体(Handle and Body)模式或接口(Interface)模式。

其实感觉 mvp架构如同桥接模式 将v层和m层隔离开了

三、模式的结构

桥接模式UML类图如下:


7017386-ce3cb4df57054e2f.png

可以看到在桥接模式的结构图中,存在一条连接两个继承等级结构的桥。

在桥接模式结构图中包含如下几个角色:

桥接模式是一个非常有用的模式,在桥接模式中体现了很多面向对象设计原则的思想,包括“单一职责原则”、“开闭原则”、“合成复用原则”、“里氏代换原则”、“依赖倒转原则”等。

在使用桥接模式时,首先应该识别出一个类所具有的两个独立变化的维度,将它们设计为两个独立的继承等级结构,为两个维度都提供抽象层,并建立抽象耦合。通常情况下,将具有两个独立变化维度的类的一些普通业务方法和与之关系最密切的维度设计为“抽象类”层次结构(抽象部分),而将另一个维度设计为“实现类”层次结构(实现部分)。

(也可以是 把一个混杂的类(因为初始的程序可能连继承都没设计) 拆成独立的两个系列)

比如上面的例子,手机可以打电话,可以玩游戏,这些时每个手机型号都具备的业务方法,所以将手机型号这一维度定为手机的抽象部分,而手机壳则是另一个维度,与手机更多是一种“设置”关系,定为手机的实现部分。

四、典型代码

一个类两个独立变化的维度,为了降低耦合度,对两个不同的维度提取抽象类和实现类接口,并建立一个抽象关联关系。对于“实现部分”维度,实现类接口典型代码如下:
(手机壳)

public interface Implementor {
    void operationImpl();
}

具体实现类中实现了在实现类接中声明的方法,典型代码如下:

public class ConcreteImplementor implements Implementor {
    public void operationImpl() {
        //todo
    }
}

另一维度,抽象类典型代码如下:
(手机)

public abstract class Abstraction {
    protected Implementor impl; //实现类接口

    public void setImpl(Implementor impl){
        this.impl = impl;
    }

    public abstract void operation();  //声明抽象业务方法
}

扩充抽象类继承抽象类,典型代码如下

public class RefinedAbstraction extends Abstraction {
    public void operation() {
        //todo
        impl.operationImpl();
        //todo
    }
}

五、代码示例

还是以引言中的手机为例:

5.1、使用多重继承

7017386-8e063a4eea8570c3.png
public abstract class Phone {
    public abstract void playMusic();
}


public class VivoPhone extends Phone {
    public void playMusic() {
        System.out.println("音乐high起来");
    }
}

public class OppoPhone extends Phone {
    public void playMusic() {
        System.out.println("音乐high起来");
    }
}

public class XiaomiPhone extends Phone {
    public void playMusic() {
        System.out.println("音乐high起来");
    }
}

public class SimpleVivoPhone extends VivoPhone {
    public void playMusic() {
        System.out.println("戴上简单手机壳");
        System.out.println("音乐high起来");
    }
}

public class CuteVivoPhone extends VivoPhone {
    public void playMusic() {
        System.out.println("戴上可爱手机壳");
        System.out.println("音乐high起来");
    }
}

public class SimpleOppoPhone extends OppoPhone {
    public void playMusic() {
        System.out.println("戴上简单手机壳");
        System.out.println("音乐high起来");
    }
}

public class CuteOppoPhone extends OppoPhone{
    public void playMusic() {
        System.out.println("戴上可爱手机壳");
        System.out.println("音乐high起来");
    }
}

public class SimpleXiaomiPhone extends XiaomiPhone {
    public void playMusic() {
        System.out.println("戴上简单手机壳");
        System.out.println("音乐high起来");
    }
}

public class CuteXiaomiPhone extends XiaomiPhone {
    public void playMusic() {
        System.out.println("戴上简单手机壳");
        System.out.println("音乐high起来");
    }
}

可以发现:

5.2、使用桥接模式

7017386-72ae7b68d21fa3ea.png

先定出抽象部分:

public abstract class Phone {
    protected ShellImplementor shellImplementor;

    public void  setShellImplementor(ShellImplementor shellImplementor){
        this.shellImplementor = shellImplementor;
    }

    public void call(){
        System.out.println("打电话");
    }

    public abstract void playMusic();
}

扩展抽象部分:

public class VivoPhone extends Phone {
    public void playMusic() {
        shellImplementor.wearShell();
        System.out.println("音乐High起来");
    }
}

public class OppoPhone extends Phone {
    public void playMusic() {
        shellImplementor.wearShell();
        System.out.println("音乐high起来");
    }
}

public class XiaomiPhone extends Phone {
    public void playMusic() {
        shellImplementor.wearShell();
        System.out.println("音乐high起来");
    }
}

将手机壳部分定义为另一个维度:

public interface ShellImplementor {
    void wearShell();
}

具体实现:

public class SimpleShell implements ShellImplementor {
    public void wearShell() {
        System.out.println("戴上简单手机壳");
    }
}

public class CuteShell implements ShellImplementor {
    public void wearShell() {
        System.out.println("戴上可爱手机壳");
    }
}

客户端测试:

public class Client {
    public static void main(String[] args) {
        Phone phone = new VivoPhone();
        ShellImplementor shell = new SimpleShell();
        phone.setShellImplementor(shell);
        phone.playMusic();
    }
}

新增手机型号只需继承自Phone就可以,新增手机壳也只需实现ShellImplementor,而且更换简单,可以用XML配置文件实现,只需修改配置,不需修改源码。

而且如果还要多一个实现,即三重维度,比如手机膜,多重继承会直接炸掉,而桥接模式则相对简洁很多。

六、优点和缺点

6.1、优点

6.2、缺点

七、适用环境

在以下情况下可以考虑使用桥接模式:

参考

设计模式--桥接模式

上一篇 下一篇

猜你喜欢

热点阅读