重读Java设计模式: 桥接模式详解

news2025/1/19 23:13:14

引言

在软件开发中,经常会遇到需要在抽象与实现之间建立连接的情况。当系统需要支持多个维度的变化时,使用传统的继承方式往往会导致类爆炸和耦合度增加的问题。为了解决这一问题,我们可以使用桥接模式。桥接模式是一种结构型设计模式,它将抽象部分与其实现部分分离,使它们可以独立地变化。本文将深入探讨桥接模式的概念、应用场景以及在实际项目中的实现方式。

一、理解桥接模式

1.1 什么是桥接模式?

桥接模式是一种结构型设计模式,旨在将抽象部分与其实现部分分离,使它们可以独立地变化。桥接模式通过将抽象和实现解耦,使得它们可以独立地扩展,而不会相互影响。这种分离使得系统更加灵活,可以应对不断变化的需求。

1.2 桥接模式的角色

在桥接模式中,通常有以下几个角色:

  • 抽象化(Abstraction):定义抽象类的接口,并维护一个指向实现化对象的引用。
  • 扩展抽象化(Refined Abstraction):对抽象化角色进行扩展,实现更加精细的抽象接口。
  • 实现化(Implementor):定义实现类的接口,供抽象化角色调用。
  • 具体实现化(Concrete Implementor):具体的实现类,实现了实现化角色定义的接口。

二、桥接模式的应用场景

2.1 多维度变化的情况

当一个类存在多个变化维度,并且这些变化维度需要独立扩展时,可以考虑使用桥接模式。例如,一个形状类有不同的颜色和绘制方式,可以使用桥接模式将形状与颜色、绘制方式分离,实现各自独立的扩展。

2.2 需要避免使用继承的情况

在传统的继承方式中,类的继承关系是静态的,一旦继承关系确定,就不容易变化。而使用桥接模式可以在运行时动态地选择抽象和实现的组合,避免了继承方式的静态性。

三、Java 中的桥接模式实现

在 Java 中,桥接模式的实现通常通过接口和实现类组合的方式来实现。我们就以上面说的一个形状有不同的颜色和绘制方式,可以使用桥接模式将形状与颜色方式分离,实现各自独立的扩展举例

先来看下示例的 UML 图:

  • 抽象化与扩展抽象化部分
public abstract class Shape {
  protected final DrawApi drawApi;

  public Shape(DrawApi api) {
    this.drawApi = api;
  }

  public abstract void draw();
}

public class Circle extends Shape {
  public Circle(DrawApi api) {
    super(api);
  }

  @Override
  public void draw() {
    System.out.println("Draw Circle : " + drawApi.draw());
  }
}

public class Square extends Shape {

    public Square(DrawApi api) {
        super(api);
    }

    @Override
    public void draw() {
        System.out.println("Draw Square: " + drawApi.draw());
    }
}
  • 实现化部分与具体实现化部分
public interface DrawApi {
  String draw();
}

public class RedDraw implements DrawApi{
  @Override
  public String draw() {
    return "使用红色颜料画图";
  }
}

public class BlueDraw implements DrawApi{
  @Override
  public String draw() {
    return "使用蓝色颜料画图";
  }
}
  • 客户端部分
package com.markus.desgin.mode.structural.bridge;

/**
 * @Author: zhangchenglong06
 * @Date: 2024/3/12
 * @Description:
 */
public class BridgePatternDemo {
  /**
   * Spring 中 JdbcTemplate 和 DataSource 之间的关系就是一种 桥接模式
   * @param args
   */
  public static void main(String[] args) {

    RedDraw redDraw = new RedDraw();
    BlueDraw blueDraw = new BlueDraw();

    Shape red = new Circle(redDraw);
    Shape blue = new Circle(blueDraw);
    red.draw();
    blue.draw();

    red = new Square(redDraw);
    blue = new Square(blueDraw);
    red.draw();
    blue.draw();
  }
}

四、桥接模式在 Spring 框架中的应用

在 Spring 框架中,使用桥接模式的一个典型例子是在 JdbcTemplateDataSource 之间。这两个组件是 Spring 框架中处理数据库操作的核心组件之一。

JdbcTemplate

JdbcTemplate 是 Spring 框架提供的一个用于简化 JDBC 编程的类,它封装了大量的 JDBC 操作,使得开发者可以通过简单的方法调用来执行数据库查询、更新等操作,而无需手动管理数据库连接、预处理语句、结果集等资源。通过使用 JdbcTemplate,开发者可以专注于业务逻辑的实现,而不用过多关注底层的数据库操作细节。

DataSource

DataSource 是 Java 中用于管理数据库连接的接口,它定义了一系列获取数据库连接的方法,如 getConnection()getConnection(username, password) 等。在实际应用中,DataSource 通常是由数据库厂商提供的具体实现,比如 Apache 的 Commons DBCP、C3P0 等。

桥接模式应用

Spring 框架中使用桥接模式的地方就是在 JdbcTemplateDataSource 之间的连接。具体来说,JdbcTemplate 充当了抽象化角色,而 DataSource 则充当了实现化角色。通过在 JdbcTemplate 中注入 DataSource 实例,Spring 可以利用桥接模式将两者连接起来,实现数据库操作的统一接口。这样一来,JdbcTemplate 就可以使用 DataSource 提供的数据库连接,从而执行相应的 SQL 操作。

下面是一个简单的示例代码,演示了 JdbcTemplateDataSource 之间的桥接模式应用:

@Configuration
public class AppConfig {

    @Bean
    public DataSource dataSource() {
        // 配置数据源,这里使用 HikariCP 数据源作为示例
        HikariConfig config = new HikariConfig();
        config.setJdbcUrl("jdbc:mysql://localhost:3306/mydatabase");
        config.setUsername("username");
        config.setPassword("password");
        return new HikariDataSource(config);
    }

    @Bean
    public JdbcTemplate jdbcTemplate(DataSource dataSource) {
        // 注入数据源到 JdbcTemplate 中
        return new JdbcTemplate(dataSource);
    }
}

在上面的示例中,我们通过 @Bean 注解配置了一个 DataSource 实例,并将其注入到 JdbcTemplate 中。这样一来,在其他组件中就可以直接注入 JdbcTemplate,然后使用它来执行数据库操作,而不用关心底层的数据库连接细节。

总之,Spring 框架中使用桥接模式将 JdbcTemplateDataSource 结合起来,使得开发者可以更加便捷地进行数据库操作,提高了开发效率和代码的可维护性。

五、设计模式百宝箱

  • 在本节,我们继续填充我们的百宝箱:

    • 面向对象基础
      • 抽象
      • 封装
      • 多态
      • 继承
    • 面向对象原则
      • 依赖抽象,不要依赖具体类
      • 针对接口编程,不针对具体实现编程
      • 类应该对扩展开放,对修改关闭
      • 为交互对象之间的松耦合设计而努力
      • 多用组合,少用继承
    • 面向对象设计模式
      • 简单工厂模式:定义了一个创建对象的接口,将创建对象的内容从客户端抽离出来
      • 抽象工厂模式:提供一个接口,用于创建相关或依赖对象的家族,而不需要明确指定具体类
      • 原型模式:通过复制现有对象来创建新对象,提高代码效率和可维护性
      • 建造者模式:将一个复杂对象的构建过程与其表示分离,使得同样的构建过程可以创建不同的表示
      • 适配器模式:将一个类的接口转换成客户期望的另一个接口。适配器让原来接口不兼容的类可以合作无间
      • 桥接模式:将抽象与实现分离,不止改变你的实现,也能改变你的抽象

六、总结

总结一下,本篇文章详细介绍了桥接模式在 Java 设计中的应用,以及在 Spring 框架中的实际应用。首先,它解释了桥接模式的基本概念和角色,以及适用场景。然后,通过一个形状与颜色绘制方式分离的示例展示了桥接模式的实现。最后,以 JdbcTemplate 和 DataSource 之间的关系为例,说明了桥接模式在 Spring 中的应用,通过桥接模式将两者连接起来,实现了数据库操作的统一接口。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1571913.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

机器学习中的GBDT模型及其优缺点(包含Python代码样例)

目录 一、简介 二、优缺点介绍 三、Python代码示例 四、总结 一、简介 GBDT(Gradient Boosting Decision Tree)是一种集成学习算法,被广泛应用于机器学习中的回归和分类问题。它由多个决策树组成,每个决策树都通过迭代逐渐提升…

巧用 STM32CubeIDE 之编译警告

1. 前言 编译警告对于工程师们来说,是再常见不过的了。对于严谨的工程师们来说,任何warning 都是不可忽视的。 2. 巧妙使用 warning 在 STM32CubeIDE 中,我们可以通过主动 warning(甚至 error)的方式来通知工程师&a…

计算机网络-TCP基础、三次挥手、四次握手过程

TCP基础 定义:TCP是面向连接的、可靠的、基于字节流的传输层通信协议。这意味着在发送数据之前,TCP需要建立连接,并且它能确保数据的可靠传输。此外,TCP将数据视为无结构的连续字节流。面向连接:TCP只能一对一进行连接…

ES6中数组新增的扩展和方法

文章目录 一、扩展运算符的应用二、构造函数新增的方法Array.from()Array.of()三、实例对象新增的方法copyWithin()find()、findIndex()fill()entries(),keys(),values()includes()flat(),flatMap()四、数组的空位五、sort排序稳定性 参考文献…

响应式网站设计哪个类型比较适合你?

响应式网站设计哪个类型比较适合你?有很多人对于响应式网站还不太了解,其实这种网站就是以创建页面的图片排版大小,来根据浏览网站用户所使用网络设备,自动化适应这些设备。 不管是出于网站兼容性还是用户信息需求,它都…

RK3568测试

作者简介: 一个平凡而乐于分享的小比特,中南民族大学通信工程专业研究生在读,研究方向无线联邦学习 擅长领域:驱动开发,嵌入式软件开发,BSP开发 作者主页:一个平凡而乐于分享的小比特的个人主页…

深入浅出 -- 系统架构之微服务架构的新挑战

尽管微服务架构有着高度独立的软件模块、单一的业务职责、可灵活调整的技术栈等优势,但也不能忽略它所带来的弊端。本篇文章,我们从网络、性能、运维、组织架构和集成测试五个方面来聊一下设计微服务架构需要考虑哪些问题,对设计有哪些挑战呢…

使用 Apipost 管理项目接口文档

目录 前言 基本使用教程 注册账号 创建项目 创建接口 编辑接口 ​编辑 接口设计 生成文档 接口用例 前言 在 web 后端开发中,开发接口之前通常需要先写接口文档,然后前后端再根据接口文档开始开发系统模块。接口文档形式多种多样&#xff0c…

Aurora8b10b(1)IP核介绍并基于IP核进行设计

文章目录 前言一、IP核设置二、基于IP核进行设计2.1、设计框图2.2、aurora_8b10b_0模块2.3、aurora_8b10b_0_CLOCK_MODULE2.4、aurora_8b10b_0_SUPPORT_RESET_LOGIC2.5、aurora8b10b_channel模块2.6、IBUFDS_GTE2模块2.7、aurora_8b10b_0_gt_common_wrapper模块2.8、aurora8b10…

基于MPPT的风力机发电系统simulink建模与仿真

目录 1.课题概述 2.系统仿真结果 3.核心程序与模型 4.系统原理简介 4.1风能与风力发电机模型 4.2风力机功率特性与最大功率点 4.3 MPPT 5.完整工程文件 1.课题概述 基于MPPT的风力机发电系统simulink建模与仿真。MPPT使用S函数编写实现。基于最大功率点跟踪&#xff08…

【OpenCV-颜色空间】

OpenCV-颜色空间 ■ RGB■ BGR■ HSV■ HSL■ HUE■ YUV ■ RGB ■ BGR BGR 就是RGB R和B调换位置。 OpenCV 默认使用BGR ■ HSV ■ HSL ■ HUE ■ YUV

免费搭建:雾锁王国Enshrouded游戏服务器,10秒完成!

免费自建雾锁王国Enshrouded服务器,先领取阿里云300元无门槛代金券,然后在雾锁王国Enshrouded专题页一键部署,不需要基础,鼠标点选即可10秒钟创建一台雾锁王国游戏服务器,超简单,阿里云服务器网aliyunfuwuq…

利用IP地址判断羊毛用户:IP数据云提供IP风险画像

在当今数字化社会,互联网已经成为人们日常生活和商业活动中不可或缺的一部分。然而,随着网络的普及,网络欺诈行为也日益猖獗,其中包括了羊毛党这一群体。羊毛党指的是利用各种手段获取利益、奖励或者优惠而频繁刷取优惠券、注册账…

leetcode热题100.跳跃游戏2

Problem: 45. 跳跃游戏 II 文章目录 题目思路复杂度Code 题目 给定一个长度为 n 的 0 索引整数数组 nums。初始位置为 nums[0]。 每个元素 nums[i] 表示从索引 i 向前跳转的最大长度。换句话说,如果你在 nums[i] 处,你可以跳转到任意 nums[i j] 处: …

【排列回溯】Leetcode 46. 全排列

【排列回溯】Leetcode 46. 全排列 ---------------🎈🎈题目链接🎈🎈------------------- used数组,其实就是记录此时temp 里都有哪些元素使用了,一个排列里一个元素只能使用一次。 class Solution {List&…

Dockerd的使用

端口映射 存储卷 类似于mount,把真机的某个目录映射都容器里面 -v 选项可以有多个 利用存储卷修改配置文件 容器间网络模式 共享网络为 --networkcontainer:容器名 微服务架构 一种由容器为载体,使用多个小型服务组合来构建复杂的架构为…

如何编写属于自己的第一个exp

0x00 前言 在我们找到一个漏洞之后,可能会想着去fofa上搜语法进而扩大战果,而有些漏洞利用起来十分繁琐,这时候就需要一个exp来批量帮我们进行扫描工作,接下来就介绍一下如何进行exp的编写,这个过程中最重要的还是体现…

12-1-CSS 常用样式属性

个人主页:学习前端的小z 个人专栏:HTML5和CSS3悦读 本专栏旨在分享记录每日学习的前端知识和学习笔记的归纳总结,欢迎大家在评论区交流讨论! 文章目录 CSS 常用样式属性1 CSS 三角形2 CSS 用户界面样式2.1 什么是界面样式2.2 鼠标…

磁盘压力测试工具(vdbenchfio)

磁盘压力测试工具(vdbench&fio) 最近有遇到对象挂载为文件系统的需求,为了测试挂载后的读写性能,有了解了一些测试工具。下面给大家分享下我使用的工具vdbench和fio。 1 vdbench 官网文档:https://www.oracle.com/…

2024/4/5 ACM格式练习

一、知识点: (1)行数不固定:用Scanf的err返回值判断是否读到EOF。 (2)每行数据个数不固定:一个一个读数据和它后面的字符,判断后面的字符是否是换行符。如果是就说明读完了一行数据…