设计模式之旅12--桥接模式

1. 定义

桥接模式也叫桥梁模式,通过抽象角色引用实现角色,将抽象和实现解耦,使得两者可以独立地变化。其中抽象和实现可以理解为两个不同的维度。

桥接模式

2. 使用场景

  • 一个类存在两个独立变化的维度,并且两个维度都需要扩展。
  • 不希望或不适用使用继承的场景。
  • 接口或抽象类不稳定的场景,明知接口不稳定还想通过实现或继承来实现业务需求,那是得不偿失的。
  • 重用性要求较高的场景,设计的颗粒度越细,则被重用的可能性就越大,而采用继承则受父类的限制,不可能出现太细的颗粒度。

3. 实现

这里以冲咖啡为例子进行代码实现,实现大杯小杯、加糖不加糖两个维度的搭配。

抽象化角色——咖啡:

/**
 * Abstraction是抽象化角色,保持实现化角色的引用
 * 这里的Abstraction代表咖啡
 */
public abstract class Abstraction {

    //抽象化角色,保持实现化角色的引用,是桥接模式的核心
    //这里的实现化角色代表咖啡的调料
    private Implementor mImplementor;

    public Abstraction(Implementor implementor) {
        mImplementor = implementor;
    }

    public Implementor getImplementor() {
        return mImplementor;
    }

    /**
     * 冲咖啡,具体实现由子类实现
     */
    public abstract void operator();

}

修正抽象化角色——大杯、小杯的咖啡:

/**
 * RefineAbstraction是修正抽象化角色,通过引用实现化角色Implementor对抽象化角色Abstraction进行修正
 * RefineAbstraction1是咖啡的实现类,代表大杯的咖啡
 */
public class RefineAbstraction1 extends Abstraction {

    public RefineAbstraction1(Implementor implementor) {
        super(implementor);
    }

    @Override
    public void operator() {
        System.out.println("大杯的" + this.getImplementor().implement() + "咖啡");
    }
}

public class RefineAbstraction2 extends Abstraction {

    public RefineAbstraction2(Implementor implementor) {
        super(implementor);
    }

    @Override
    public void operator() {
        System.out.println("小杯的" + this.getImplementor().implement() + "咖啡");
    }
}

实现化角色——调料:

/**
 * Implementor是实现化角色
 * 这里代表咖啡的调料,具体调料由实现类实现
 */
public interface Implementor {

    String implement();

}

具体实现化角色——加糖、不加糖:

/**
 * ConcreteImplementor具体的实现化角色
 * ConcreteImplementor咖啡的调料的实现类,代表加糖
 * 
 */
public class ConcreteImplementor1 implements Implementor {
    @Override
    public String implement() {
        return "加糖";
    }
}

public class ConcreteImplementor2 implements Implementor {
    @Override
    public String implement() {
        return "不加糖";
    }
}

场景类:

public class Client {

    public static void main(String[] args) {

        Implementor i1 = new ConcreteImplementor1();//加糖
        Implementor i2 = new ConcreteImplementor1();//不加糖

        Abstraction a1 = new RefineAbstraction1(i1);//大杯、加糖
        Abstraction a2 = new RefineAbstraction1(i2);//大杯、不加糖
        Abstraction a3 = new RefineAbstraction2(i1);//小杯、加糖
        Abstraction a4 = new RefineAbstraction2(i2);//小杯、不加糖

        a1.operator();
        a2.operator();
        a3.operator();
        a4.operator();
    }

}

运行结果:

大杯的加糖咖啡
大杯的加糖咖啡
小杯的加糖咖啡
小杯的加糖咖啡

4. 优点

  • 抽象和实现分离。解决了继承的缺点,桥接模式实现了不受抽象的约束,不用再绑定在一个固定的抽象层次上。
  • 优秀的扩充能力。抽象和实现都可以独立扩展。
  • 实现细节对客户透明。客户不用关心细节的实现,它已经由抽象层通过聚合关系完成了封装。

5. 缺点

  • 桥接模式的引入会增加系统的理解与设计难度。由于聚合关联关系建立在抽象层,要求开发者针对抽象设计和编程。
  • 实际中不容易发现可以使用桥接模式的地方。桥接模式要求正确识别出系统中两个独立变化的维度,因此其使用范围具有一定的局限性。

你可能感兴趣的:(设计模式之旅12--桥接模式)