设计模式--- 桥接模式、JDBC 源码剖析(桥接)

本文涉及的产品
云数据库 RDS MySQL,集群版 2核4GB 100GB
推荐场景:
搭建个人博客
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
RDS MySQL Serverless 高可用系列,价值2615元额度,1个月
简介: 设计模式--- 桥接模式、JDBC 源码剖析(桥接)

1、手机操作问题

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

image.png

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

类图

image.png

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

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

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

解决方案 --> 使用桥接模式

3、桥接模式基本介绍

桥接模式(Bridge模式)是指:将实现与抽象放在两个不同的类层次中,使两个层次可以独立改变,桥接模式是一种结构型设计模式

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

image.png

Client:桥接模式的调用者
Abstraction:抽象类,Abstraction 中维护了一个 Implementor 实现类的实例(聚合关系),Abstraction 充当桥接类
RefinedAbstraction:Abstraction 的具体实现类
Implementor:定义行为的接口
ConcreteImplementor:Implementor 的具体实现类
从 UML 图: 这里的抽象类和接口是聚合的关系, 其实也是调用和被调用关系,抽象在 Abstraction 这一块,行为实现在 Implementor 这一块

5、桥接模式解决手机问题

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

类图

image.png

1.Brand:规定各个品牌手机的行为

代码实现

public interface Brand {
    void open();
    void close();
    void call();
}

2.XiaoMi:实现了 Brand 接口,指定了小米手机的具体行为

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(" 小米手机打电话 ");

    }
}

3.Vivo:实现了 Brand 接口,指定了 Vivo 手机的具体行为

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手机打电话 ");

    }
}

4.Phone:电话的抽象类,在该类中聚合了一个 Brand 接口的具体实现类

public abstract class Phone {
    //聚合品牌
    private Brand brand;
    //构造器
    public Phone(Brand brand){
//        super();
        this.brand=brand;
    }
    protected void open(){
        this.brand.open();
    }
    protected void close(){
        this.brand.close();
    }
    protected void call(){
        this.brand.call();
    }
}

5.FoldedPhone:继承抽象父类 Phone,对抽象父类中的方法进行重写

package java设计模式.Bridge;
//抽象类调用的是成员接口的实现类的内容
//折叠式手机类,继承 抽象类 Phone
public class FoldedPhone extends Phone{
    // 构造器
    public FoldedPhone(Brand brand) {
        //父类的构造器
        super(brand);
    }
    public void open(){
        super.open();
        System.out.println("折叠样式手机");
    }

    @Override
    public void close() {
        super.close();
        System.out.println(" 折叠样式手机 ");
    }

    @Override
    public void call() {
        super.call();
        System.out.println(" 折叠样式手机 ");
    }

}

6.FoldedPhone:继承抽象父类 Phone,对抽象父类中的方法进行重写

package java设计模式.Bridge;

public class UpRightPhone extends Phone{
    // 构造器
    public UpRightPhone(Brand brand) {
        super(brand);
    }

    @Override
    public void open() {
        super.open();
        System.out.println(" 直立样式手机 ");
    }

    @Override
    public void close() {
        super.close();
        System.out.println(" 直立样式手机 ");
    }

    @Override
    public void call() {
        super.call();
        System.out.println(" 直立样式手机 ");
    }

}

7.Client:客户端,可以看到,使用桥接模式可以轻松地组合出不同手机类型、不同品牌的手机

package java设计模式.Bridge;

public class Client {
    public static void main(String[] args) {
        // 获取折叠式的小米手机 (样式 + 品牌 )
        //父类的变量指向子类的实例
        Phone phone1= new FoldedPhone(new XiaoMi());
        phone1.open();
        phone1.close();
        phone1.close();
        Phone phone2= new FoldedPhone(new Vivo());
        phone2.open();
        phone2.close();
        phone2.close();
        UpRightPhone upRightPhone = new UpRightPhone(new XiaoMi());
        upRightPhone.open();
        upRightPhone.close();
        upRightPhone.call();
    }
}

总结

Phone 就像一座桥的感觉,它其实并没有做什么实质性的工作,只是调用 Brand 的具体实现类中的方法,就感觉像是一个请求从 Phone 的具体实现类通过 Phone 传递到了 Brand 的具体实现类
通过代码我们可以看到,增加一个新的手机样式,并不会引起类膨胀,因为只要新样式继承了 Phone,并通过构造器或者 setter 方法聚合一个 Brand 实现类的实例,就能完成组合的作用

JDBC 源码剖析

JdbcDriver接口,如果从桥接模式来看, Driver就是一个接口(行为规范),下面可以有MySQLDriverOracleDriver,这些就可以当做实现接口类

类图

image.png

代码追踪

  1. 客户端通过 DriverManager 操作数据库,DriverManager 里面定义了很多方法,就比如说如下的 getConnection() 方法,它返回一个 Connection 对象

image.png

@CallerSensitive
public static Connection getConnection(String url,
    java.util.Properties info) throws SQLException {

    return (getConnection(url, info, Reflection.getCallerClass()));
  
}

2.Connectionjava.sql 包下的接口,里面定义了 超多的抽象方法,比如 prepareStatement() 方法

public interface Connection  extends Wrapper, AutoCloseable {
    
    // ...  
    PreparedStatement prepareStatement(String sql, int resultSetType,
                                       int resultSetConcurrency, int resultSetHoldability)
        throws SQLException;

    CallableStatement prepareCall(String sql, int resultSetType,
                                  int resultSetConcurrency,
                                  int resultSetHoldability) throws SQLException;

    PreparedStatement prepareStatement(String sql, int autoGeneratedKeys)
        throws SQLException;

    PreparedStatement prepareStatement(String sql, int columnIndexes[])
        throws SQLException;

    PreparedStatement prepareStatement(String sql, String columnNames[])
        throws SQLException;

    // ...

3.com.mysql.jdbc.Connection 接口继承了 java.sql.Connection 接口

image.png

public interface Connection extends java.sql.Connection, ConnectionProperties {

4.ConnectionImpl 类实现了 MySQLConnection 接口,其中 MySQLConnection 接口继承了 com.mysql.jdbc.Connection 接口

public class ConnectionImpl extends ConnectionPropertiesImpl implements MySQLConnection {
相关实践学习
基于Hologres轻松玩转一站式实时仓库
本场景介绍如何利用阿里云MaxCompute、实时计算Flink和交互式分析服务Hologres开发离线、实时数据融合分析的数据大屏应用。
SaaS 模式云数据仓库必修课
本课程由阿里云开发者社区和阿里云大数据团队共同出品,是SaaS模式云原生数据仓库领导者MaxCompute核心课程。本课程由阿里云资深产品和技术专家们从概念到方法,从场景到实践,体系化的将阿里巴巴飞天大数据平台10多年的经过验证的方法与实践深入浅出的讲给开发者们。帮助大数据开发者快速了解并掌握SaaS模式的云原生的数据仓库,助力开发者学习了解先进的技术栈,并能在实际业务中敏捷的进行大数据分析,赋能企业业务。 通过本课程可以了解SaaS模式云原生数据仓库领导者MaxCompute核心功能及典型适用场景,可应用MaxCompute实现数仓搭建,快速进行大数据分析。适合大数据工程师、大数据分析师 大量数据需要处理、存储和管理,需要搭建数据仓库?学它! 没有足够人员和经验来运维大数据平台,不想自建IDC买机器,需要免运维的大数据平台?会SQL就等于会大数据?学它! 想知道大数据用得对不对,想用更少的钱得到持续演进的数仓能力?获得极致弹性的计算资源和更好的性能,以及持续保护数据安全的生产环境?学它! 想要获得灵活的分析能力,快速洞察数据规律特征?想要兼得数据湖的灵活性与数据仓库的成长性?学它! 出品人:阿里云大数据产品及研发团队专家 产品 MaxCompute 官网 https://www.aliyun.com/product/odps 
相关文章
|
1月前
|
设计模式 开发者
【设计模式】第七篇:和我一起简单认识桥接模式
实现的意思并不是指抽象的派生类,而是指通过组合来代替继承关系,从而降低抽象和具体实现产品两个可变换维度之间的耦合,就像我们的相机品牌和相机产品类型之间的分离
27 4
|
2月前
|
设计模式 Java
【设计模式】JAVA Design Patterns——Bridge(桥接模式)
【设计模式】JAVA Design Patterns——Bridge(桥接模式)
【设计模式】JAVA Design Patterns——Bridge(桥接模式)
|
13天前
|
设计模式 JavaScript
js设计模式【详解】—— 桥接模式
js设计模式【详解】—— 桥接模式
26 6
|
19天前
|
设计模式
桥接模式-大话设计模式
桥接模式-大话设计模式
12 1
|
17天前
|
设计模式 Java 数据库
Java设计模式:桥接模式实现灵活组合,超越单一继承的设计之道(十)
Java设计模式:桥接模式实现灵活组合,超越单一继承的设计之道(十)
|
19天前
|
设计模式 Java
Java设计模式之桥接模式详解
Java设计模式之桥接模式详解
|
2月前
|
设计模式 API
【设计模式】适配器和桥接器模式有什么区别
【设计模式】适配器和桥接器模式有什么区别
46 1
|
2月前
|
设计模式 安全 Java
【初学者慎入】Spring源码中的16种设计模式实现
以上是威哥给大家整理了16种常见的设计模式在 Spring 源码中的运用,学习 Spring 源码成为了 Java 程序员的标配,你还知道Spring 中哪些源码中运用了设计模式,欢迎留言与威哥交流。
|
2月前
|
设计模式 Java Go
[设计模式Java实现附plantuml源码~结构型]不兼容结构的协调——适配器模式
[设计模式Java实现附plantuml源码~结构型]不兼容结构的协调——适配器模式
|
2月前
|
设计模式 算法 Java
[设计模式Java实现附plantuml源码~行为型]定义算法的框架——模板方法模式
[设计模式Java实现附plantuml源码~行为型]定义算法的框架——模板方法模式