1、手机操作问题

现在对不同手机类型的不同品牌实现操作编程(比如:开机、关机、上网、打电话等),如图:

ys50y

2、传统方案解决手机操作问题

思路分析(类图)

962o7

3、传统方案解决手机操作问题分析

1)扩展性问题(类爆炸),如果我们再增加手机的样式(旋转式),就需要增加各个品牌手机的类,同样如果我们增加一个手机品牌,也要在各个手机样式下增加。

2)违反了单一职责原则,当我们增加手机样式时,要同时增加所有品牌的手机,这样增加了代码维护成本。

3)解决方案 => 使用桥接模式

4、桥接模式

4.1 基本介绍

1)桥接模式(Bridge 模式)是指:将实现与抽象放在两个不同的类层次中,使两个层次可以独立改变。

2)是一种结构型设计模式

3)Bridge 模式基于类的最小设计原则,通过使用封装、聚合及继承等行为让不同的类承担不同的职责。它的主要特点是把抽象(Abstraction)与行为实现(Implementation)分离开来,从而可以保持各部分的独立性以及应对他们的功能扩展。

4.2 原理类图

6m2cq

原理类图说明:

1)Client 类:桥接模式的调用者

2)抽象类(Abstraction):维护了 Implementor / 即它的实现类,ConcreteImplementorA...,二者是聚合关系,Abstraction 充当桥接类

3)RefinedAbstraction:是 Abstraction 抽象类的子类

4)Implementor:行为实现类的接口

5)ConcreteImplementorA/B:行为的具体实现类

6)从 UML 图:这里的抽象类和接口是聚合的关系,其实调用和被调用关系

4.3 应用实例

使用桥接模式改进传统方式,让程序具有更好的扩展性,利用程序维护

1)应用实例说明:

现在对不同手机类型的不同品牌实现操作编程(比如:开机、关机、上网、打电话等),如图:

sxdtc

2)思路分析-类图:

vkybf

3)代码实现

Brand 品牌接口

package com.wenze.design.bridge;  
  
/**  
 * 品牌  
 *  
 * @author wenze  
 * @version 1.0  
 * @Date 2023-10-25 16:17:04  
 * @since 1.0  
 */
public interface Brand {  
    /**  
     * 开机  
     */  
    void open();  
    /**  
     * 关机  
     */  
    void close();  
    /**  
     * 打电话  
     */  
    void call();  
}

XiaoMi 小米品牌类

package com.wenze.design.bridge;  
  
/**  
 * 小米手机  
 *  
 * @author wenze  
 * @version 1.0  
 * @Date 2023-10-25 16:18:00  
 * @since 1.0  
 */
public class XiaoMi implements Brand {  
    /**  
     * 开机  
     */  
    @Override  
    public void open() {  
        System.out.println("小米手机开机");  
    }  
    /**  
     * 关机  
     */  
    @Override  
    public void close() {  
        System.out.println("小米手机关机");  
    }  
    /**  
     * 打电话  
     */  
    @Override  
    public void call() {  
        System.out.println("小米手机打电话");  
    }  
}

Vivo vivo品牌类

package com.wenze.design.bridge;  
  
/**  
 * vivo 手机  
 *  
 * @author wenze  
 * @version 1.0  
 * @Date 2023-10-25 16:18:56  
 * @since 1.0  
 */
public class Vivo implements Brand {  
    /**  
     * 开机  
     */  
    @Override  
    public void open() {  
        System.out.println("Vivo手机开机");  
    }  
    /**  
     * 关机  
     */  
    @Override  
    public void close() {  
        System.out.println("Vivo手机关机");  
    }  
    /**  
     * 打电话  
     */  
    @Override  
    public void call() {  
        System.out.println("Vivo手机打电话");  
    }  
}

Phone 手机抽象类

package com.wenze.design.bridge;  
  
/**  
 * 手机抽象类  
 *  
 * @author wenze  
 * @version 1.0  
 * @Date 2023-10-25 16:21:51  
 * @since 1.0  
 */
public abstract class Phone {  
    private Brand brand;  
    public Phone(final Brand brand) {  
        this.brand = brand;  
    }  
    public void open() {  
        brand.open();  
    }  
    public void close() {  
        brand.close();  
    }  
    public void call() {  
        brand.call();  
    }  
}

FoldedPhone 折叠手机类

package com.wenze.design.bridge;  
  
/**  
 * 折叠式手机  
 *  
 * @author wenze  
 * @version 1.0  
 * @Date 2023-10-25 16:23:12  
 * @since 1.0  
 */
public class FoldedPhone extends Phone {  
    public FoldedPhone(final Brand brand) {  
        super(brand);  
    }  
    @Override  
    public void open() {  
        System.out.print("折叠式");  
        super.open();  
    }  
    @Override  
    public void close() {  
        System.out.print("折叠式");  
        super.close();  
    }
    @Override  
    public void call() {  
        System.out.print("折叠式");  
        super.call();  
    }  
}

UpRightPhone 直立手机类

package com.wenze.design.bridge;  
  
/**  
 * 直立式手机  
 *  
 * @author wenze  
 * @version 1.0  
 * @Date 2023-10-25 16:28:51  
 * @since 1.0  
 */
public class UpRightPhone extends Phone {  
    public UpRightPhone(final Brand brand) {  
        super(brand);  
    }  
    @Override  
    public void open() {  
        System.out.print("直立式");  
        super.open();  
    }
    @Override  
    public void close() {  
        System.out.print("直立式");  
        super.close();  
    }
    @Override  
    public void call() {  
        System.out.print("直立式");  
        super.call();  
    }  
}

Client 客户端

package com.wenze.design.bridge;  
  
/**  
 * 客户端  
 *  
 * @author wenze  
 * @version 1.0  
 * @Date 2023-10-25 16:27:22  
 * @since 1.0  
 */
public class Client {  
    public static void main(String[] args) {  
        Phone phone1 = new FoldedPhone(new XiaoMi());  
        phone1.open();  
        phone1.call();  
        phone1.close();  
  
        System.out.println("==========================");  
  
        Phone phone2 = new FoldedPhone(new Vivo());  
        phone2.open();  
        phone2.call();  
        phone2.close();  
  
        System.out.println("==========================");  
  
        Phone phone3 = new UpRightPhone(new XiaoMi());  
        phone3.open();  
        phone3.call();  
        phone3.close();  
  
        System.out.println("==========================");  
  
        Phone phone4 = new UpRightPhone(new Vivo());  
        phone4.open();  
        phone4.call();  
        phone4.close();  
    }  
}

5、桥接模式在 JDBC 源码剖析

1)JDBC 的 Driver 接口,如果从桥接模式来看,Driver 就是一个接口,下面可以有 MySQL的 Driver,Oracle 的 Driver,这些就可以当做实现接口类

2)代码分析 + Debug 源码

ytf87

lec3h

说明

  1. MySQL有自己的 ConnectionImpl 类,同样 Oracle 也有对应的实现类
  2. Driver 和 Connection 之间是通过 DriverManager 类进行桥接的

6、桥接模式的注意事项和细节

1)实现了抽象和实现部分的分离,从而极大地提供了系统的灵活性,让抽象部分和实现部分独立开来,这有助于系统进行分层设计,从而产生更好的结构化系统。

2)对于系统的高层部分,只需要知道抽象部分和实现部分的接口就可以了,其他的部分由具体业务来完成。

3)桥接模式替代多层继承方案,可以减少子类的个数,降低系统的管理和维护成本。

4)桥接模式的引入增加了系统的理解和设计难度,由于聚合关联关系建立在抽象层,要求开发者针对抽象进行设计和编程。

5)桥接模式要求正确识别出系统中两个独立变化的维度(抽象和实现),因此其使用范围有一定的局限性,即需要有这样的应用场景。

7、桥接模式其他应用场景

1)对于那些不希望使用继承或因为多层次继承导致系统类的个数急剧增加的系统,桥接模式尤为适用。

2)常见的应用场景:

  • JDBC 驱动程序
  • 银行转账系统:
    • 转账分类:网上转账、柜台转账、ATM 转账
    • 转账用户类型:普通用户、银卡用户、金卡用户
  • 消息管理:
    • 消息类型:即时消息、延时消息
    • 消息分类:手机短信、邮件消息、QQ消息