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

本文涉及的产品
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS PostgreSQL,集群系列 2核4GB
简介: 设计模式--- 桥接模式、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 {
相关实践学习
基于MaxCompute的热门话题分析
本实验围绕社交用户发布的文章做了详尽的分析,通过分析能得到用户群体年龄分布,性别分布,地理位置分布,以及热门话题的热度。
SaaS 模式云数据仓库必修课
本课程由阿里云开发者社区和阿里云大数据团队共同出品,是SaaS模式云原生数据仓库领导者MaxCompute核心课程。本课程由阿里云资深产品和技术专家们从概念到方法,从场景到实践,体系化的将阿里巴巴飞天大数据平台10多年的经过验证的方法与实践深入浅出的讲给开发者们。帮助大数据开发者快速了解并掌握SaaS模式的云原生的数据仓库,助力开发者学习了解先进的技术栈,并能在实际业务中敏捷的进行大数据分析,赋能企业业务。 通过本课程可以了解SaaS模式云原生数据仓库领导者MaxCompute核心功能及典型适用场景,可应用MaxCompute实现数仓搭建,快速进行大数据分析。适合大数据工程师、大数据分析师 大量数据需要处理、存储和管理,需要搭建数据仓库?学它! 没有足够人员和经验来运维大数据平台,不想自建IDC买机器,需要免运维的大数据平台?会SQL就等于会大数据?学它! 想知道大数据用得对不对,想用更少的钱得到持续演进的数仓能力?获得极致弹性的计算资源和更好的性能,以及持续保护数据安全的生产环境?学它! 想要获得灵活的分析能力,快速洞察数据规律特征?想要兼得数据湖的灵活性与数据仓库的成长性?学它! 出品人:阿里云大数据产品及研发团队专家 产品 MaxCompute 官网 https://www.aliyun.com/product/odps 
相关文章
|
6月前
|
设计模式 开发者
【设计模式】第七篇:和我一起简单认识桥接模式
实现的意思并不是指抽象的派生类,而是指通过组合来代替继承关系,从而降低抽象和具体实现产品两个可变换维度之间的耦合,就像我们的相机品牌和相机产品类型之间的分离
54 4
|
7月前
|
设计模式 Java
【设计模式】JAVA Design Patterns——Bridge(桥接模式)
【设计模式】JAVA Design Patterns——Bridge(桥接模式)
【设计模式】JAVA Design Patterns——Bridge(桥接模式)
|
2月前
|
设计模式 PHP 开发者
PHP中的设计模式:桥接模式的解析与应用
在软件开发的浩瀚海洋中,设计模式如同灯塔一般,为开发者们指引方向。本文将深入探讨PHP中的一种重要设计模式——桥接模式。桥接模式巧妙地将抽象与实现分离,通过封装一个抽象的接口,使得实现和抽象可以独立变化。本文将阐述桥接模式的定义、结构、优缺点及其应用场景,并通过具体的PHP示例代码展示如何在实际项目中灵活运用这一设计模式。让我们一起走进桥接模式的世界,感受它的魅力所在。
|
3月前
|
设计模式 Java 关系型数据库
【Java笔记+踩坑汇总】Java基础+JavaWeb+SSM+SpringBoot+SpringCloud+瑞吉外卖/谷粒商城/学成在线+设计模式+面试题汇总+性能调优/架构设计+源码解析
本文是“Java学习路线”专栏的导航文章,目标是为Java初学者和初中高级工程师提供一套完整的Java学习路线。
452 37
|
3月前
|
设计模式 自然语言处理 算法
PHP中的设计模式:桥接模式的深入探索与应用
在PHP开发中,理解并运用设计模式是提升代码质量与可维护性的关键。本文聚焦于桥接模式——一种结构型设计模式,它通过封装一个抽象的接口,将实现与抽象分离,从而使得它们可以独立变化。不同于传统摘要的概述式表述,本文将以故事化的情境引入,逐步解析桥接模式的精髓,通过PHP代码示例详细展示其在实际项目中的应用,旨在为读者提供一个既深刻又易于理解的学习体验。
27 1
|
3月前
|
设计模式 Java
Java设计模式-桥接模式(9)
Java设计模式-桥接模式(9)
|
2月前
|
设计模式 Java
Java设计模式之桥接模式
这篇文章介绍了Java设计模式中的桥接模式,包括桥接模式的目的、实现方式,并通过具体代码示例展示了如何分离抽象与实现,使得两者可以独立变化。
46 0
|
4月前
|
设计模式 XML 存储
【七】设计模式~~~结构型模式~~~桥接模式(Java)
文章详细介绍了桥接模式(Bridge Pattern),这是一种对象结构型模式,用于将抽象部分与实现部分分离,使它们可以独立地变化。通过实际的软件开发案例,如跨平台视频播放器的设计,文章阐述了桥接模式的动机、定义、结构、优点、缺点以及适用场景,并提供了完整的代码实现和测试结果。桥接模式适用于存在两个独立变化维度的系统,可以提高系统的可扩展性和灵活性。
【七】设计模式~~~结构型模式~~~桥接模式(Java)
|
4月前
|
设计模式 存储 缓存
Guava 源码中7种设计模式的实现分析
V 哥在学习 Guava 源码中总结的7个设计模式的实现分析,欢迎关注威哥爱编程,做自己的技术,让别人去卷吧。
|
4月前
|
设计模式 缓存 项目管理
设计模式的基础问题之桥接模式在软件开发应用的问题如何解决
设计模式的基础问题之桥接模式在软件开发应用的问题如何解决