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

引言

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

一、理解桥接模式

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 中的应用,通过桥接模式将两者连接起来,实现了数据库操作的统一接口。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

计算机网络—HTTP协议:深入解析与应用实践

​ 🎬慕斯主页:修仙—别有洞天 ♈️今日夜电波:ヒステリックナイトガール 1:03━━━━━━️💟──────── 5:06 🔄 ◀️ ⏸ ▶️ ☰…

vue 实现的h5 页面,如何设置页面中的 title

修改页面中的title 公共修改方式在App.vue 中: created() {document.title "测试标题"; },单个页面修改,就在单个页面编写就ok

二百二十九、离线数仓——离线数仓Hive从Kafka、MySQL到ClickHouse的完整开发流程

一、目的 为了整理离线数仓开发的全流程,算是温故知新吧 离线数仓的数据源是Kafka和MySQL数据库,Kafka存业务数据,MySQL存维度数据 采集工具是Kettle和Flume,Flume采集Kafka数据,Kettle采集MySQL数据 离线数仓是Hi…

JVM专题——垃圾回收

本文部分内容节选自Java Guide和《深入理解Java虚拟机》, Java Guide地址: https://javaguide.cn/java/jvm/jvm-garbage-collection.html 🚀 基础(上) → 🚀 基础(中) → 🚀基础(下&…

2024-04-05 问AI: 介绍一下深度学习中的Leaky ReLU函数

文心一言 Leaky ReLU(Leaky Rectified Linear Unit)函数是深度学习领域中的一种激活函数,主要用于解决ReLU(Rectified Linear Unit)函数在负值区域为零的问题。ReLU函数在输入为正时保持原样,而在输入为负…

(学习日记)2024.04.07:UCOSIII第三十五节:互斥量实验

写在前面: 由于时间的不足与学习的碎片化,写博客变得有些奢侈。 但是对于记录学习(忘了以后能快速复习)的渴望一天天变得强烈。 既然如此 不如以天为单位,以时间为顺序,仅仅将博客当做一个知识学习的目录&a…

通过 Cookie、Redis共享Session 和 Spring 拦截器技术,实现对用户登录状态的持有和清理(三)

本篇内容对应 “2.4 生成验证码” 小节 和 “4.7 优化登陆模块”小节 视频链接 1 Kaptcha介绍 Kaotcga是一个生成验证码的工具。 你的网站验证码是什么? 在我们这个牛客论坛项目,验证码分为两部分 给用户看的是图片,用户根据图片上显示的…

跨境电商独立站是什么?为什么要做独立站?

跨境电商独立站就是跨境电商自行搭建的销售网站,服务器、域名都是自主购买的,并由跨境电商独立运营与营销推广。 近些年来,各类第三方电商平台虽然流量大,但是随着进驻电商数量的增加,流量竞争也愈发激烈,…

基于顺序表实现通讯管理系统!(有完整源码!)

​​​​​​​ 个人主页:秋风起,再归来~ 文章专栏:C语言实战项目 个人格言:悟已往之不谏,知来者犹可追 克心守己,律己则安!​​​​​​​ 目录 1、实现思路 ​…

C语言中strlen函数的实现

C语言中strlen函数的实现 为了便于和strlen函数区别,以下命令为_strlen。 描述:实现strlen,获取字符串的长度,函数原型如下: size_t strlen(const char *str);_strlen实现: size_t _strlen(const char*…

彩虹聚合DNS管理系统,附带系统搭建教程

聚合DNS管理系统,可以实现在一个网站内管理多个平台的域名解析,目前已支持的域名平台有:阿里云、腾讯云、华为云、西部数码、CloudFlare。 本系统支持多用户,每个用户可分配不同的域名解析权限;支持API接口&#xff0…

武汉星起航:跨境电商领域的领航者,助力全球贸易新篇章

自2017年以来,武汉星起航一直专注于亚马逊自营店铺,积累了宝贵的经验。2020年正式成立后,公司以跨境电商为核心,致力于为合作伙伴提供深入的合作模式。武汉星起航凭借其卓越的服务和实战经验,已成功助力众多创业者实现…

基于SpringBoot的“智慧外贸平台”的设计与实现(源码+数据库+文档+PPT)

基于SpringBoot的“智慧外贸平台”的设计与实现(源码数据库文档PPT) 开发语言:Java 数据库:MySQL 技术:SpringBoot 工具:IDEA/Ecilpse、Navicat、Maven 系统展示 系统总体结构图 平台首页界面图 商品信息界面图 …

Java8 进阶

Java8 进阶 文章目录 Java8 进阶什么是函数式接口&#xff1f;public interface Supplierpublic interface Consumerpublic interface Predicatepublic interface FunctionJava8 特性总结&#xff1a;一、Function<T, R>二、Consumer<T>三、Supplier<T>四、P…

位运算-191. 位1的个数- 136. 只出现一次的数字

位1的个数 已解答 简单 相关标签 相关企业 编写一个函数&#xff0c;输入是一个无符号整数&#xff08;以二进制串的形式&#xff09;&#xff0c;返回其二进制表达式中 设置位 的个数&#xff08;也被称为汉明重量&#xff09;。 示例 1&#xff1a; 输入&#xff1a;n 11 输…

Linux第4课 Linux的基本操作

文章目录 Linux第4课 Linux的基本操作一、图形界面介绍二、终端界面介绍 Linux第4课 Linux的基本操作 一、图形界面介绍 本节以Ubuntu系统的GUI为例进行说明&#xff0c;Linux其他版本可自行网搜。 图形系统进入后&#xff0c;左侧黄框内为菜单栏&#xff0c;右侧为桌面&…

c# 指数搜索(Exponential Search)

该搜索算法的名称可能会产生误导&#xff0c;因为它的工作时间为 O(Log n)。该名称来自于它搜索元素的方式。 给定一个已排序的数组和要 搜索的元素 x&#xff0c;找到 x 在数组中的位置。 输入&#xff1a;arr[] {10, 20, 40, 45, 55} x 45 输出&#xff1a;在索…

检验平台最基本的技术要求有哪几条

检验平台最基本的技术要求通常有以下几条&#xff1a; 系统稳定性&#xff1a;检验平台应具备良好的稳定性&#xff0c;能够长时间运行而不出现系统崩溃或异常情况。 数据安全性&#xff1a;检验平台应具备对数据进行安全存储和传输的能力&#xff0c;确保数据不被非法获取、篡…

吴恩达机器学习笔记:第 6 周-11机器学习系统的设计(Machine Learning System Design)11.1-11.5

目录 第 6 周 11、 机器学习系统的设计(Machine Learning System Design)11.1 首先要做什么11.2 误差分析11.3 类偏斜的误差度量11.4 查准率和查全率之间的权衡11.5 机器学习的数据 第 6 周 11、 机器学习系统的设计(Machine Learning System Design) 11.1 首先要做什么 在接…

基于Python的豆瓣电影评分可视化,豆瓣电影评分预测系统

博主介绍&#xff1a;✌程序员徐师兄、7年大厂程序员经历。全网粉丝12w、csdn博客专家、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精彩专栏推荐订阅&#x1f447;…