桥接模式详解和分析JDBC中的应用

🎯 设计模式专栏,持续更新中, 欢迎订阅:JAVA实现设计模式
🛠️ 希望小伙伴们一键三连,有问题私信都会回复,或者在评论区直接发言

桥接模式

文章目录

  • 桥接模式
    • 桥接模式的四个核心组成:
    • 案例:电视与遥控器的桥接模式
    • ⚙️代码实现
      • Step 1: 实现 `TV` 接口 (Implementor)
      • Step 2: 创建 `SonyTV` 和 `SamsungTV` 类 (Concrete Implementor)
      • Step 3: 创建 `RemoteControl` 类 (Abstraction)
      • Step 4: 创建 `SonyRemoteControl` 和 `SamsungRemoteControl` (Refined Abstraction)
      • 🌟 Main函数展示桥接模式的使用
    • 源码应用
      • JDBC中的桥接模式源码探索:
      • JDBC桥接模式总结:
    • 📜 总结:

桥接模式(Bridge Pattern)是一种结构型设计模式,目的是将抽象部分与它的实现部分分离开来,使它们可以独立变化。这种模式通过组合的方式(而不是继承)将不同的维度分离,使代码更加灵活、易于扩展。

简单说法: 想象你有一个电视遥控器,遥控器负责打开/关闭电视、调节音量等操作。而不同品牌的电视(如索尼、三星)可能实现这些操作的方式不一样。桥接模式允许你在不修改遥控器代码的情况下,切换或扩展不同品牌的电视。遥控器和电视品牌之间使用接口桥接,使得它们可以独立变化。

桥接模式的四个核心组成:

  1. Abstraction(抽象部分):高层接口,定义抽象的操作,如遥控器的功能(打开、关闭、调音量等)。
  2. Refined Abstraction(扩展抽象部分):扩展自Abstraction,定义更具体的遥控器类型。
  3. Implementor(实现部分):具体实现接口,不同的实现类可以完成具体操作(不同品牌电视的功能实现)。
  4. Concrete Implementor(具体实现部分):Implementor的具体实现,如SonyTV和SamsungTV。

案例:电视与遥控器的桥接模式

你有一个通用的遥控器,可以操作不同品牌的电视,遥控器定义了on()off()setChannel()等功能,而每个品牌的电视实现这些功能的细节可能不同。通过桥接模式,我们将遥控器和电视品牌分离,便于扩展。

UML类图结构

Abstraction                    ImplementorRemoteControl   ---->    TV/     \                    /    \
Concrete    SonyRemote   SonyTV  SamsungTV
Abstraction                ConcreteImplementor

⚙️代码实现

Step 1: 实现 TV 接口 (Implementor)

// TV接口定义基本操作
interface TV {void on();void off();void setChannel(int channel);
}

Step 2: 创建 SonyTVSamsungTV 类 (Concrete Implementor)

// SonyTV 具体实现类
class SonyTV implements TV {@Overridepublic void on() {System.out.println("Sony TV is now ON");}@Overridepublic void off() {System.out.println("Sony TV is now OFF");}@Overridepublic void setChannel(int channel) {System.out.println("Sony TV: Channel set to " + channel);}
}// SamsungTV 具体实现类
class SamsungTV implements TV {@Overridepublic void on() {System.out.println("Samsung TV is now ON");}@Overridepublic void off() {System.out.println("Samsung TV is now OFF");}@Overridepublic void setChannel(int channel) {System.out.println("Samsung TV: Channel set to " + channel);}
}

Step 3: 创建 RemoteControl 类 (Abstraction)

// 遥控器抽象类,组合TV接口
abstract class RemoteControl {protected TV tv;// 通过构造方法传递TV对象,形成桥接public RemoteControl(TV tv) {this.tv = tv;}public abstract void on();public abstract void off();public abstract void setChannel(int channel);
}

Step 4: 创建 SonyRemoteControlSamsungRemoteControl (Refined Abstraction)

// 扩展的具体遥控器类,使用不同的TV实现
class SonyRemoteControl extends RemoteControl {public SonyRemoteControl(TV tv) {super(tv);}@Overridepublic void on() {System.out.println("Using Sony Remote Control:");tv.on();}@Overridepublic void off() {System.out.println("Using Sony Remote Control:");tv.off();}@Overridepublic void setChannel(int channel) {System.out.println("Using Sony Remote Control:");tv.setChannel(channel);}
}class SamsungRemoteControl extends RemoteControl {public SamsungRemoteControl(TV tv) {super(tv);}@Overridepublic void on() {System.out.println("Using Samsung Remote Control:");tv.on();}@Overridepublic void off() {System.out.println("Using Samsung Remote Control:");tv.off();}@Overridepublic void setChannel(int channel) {System.out.println("Using Samsung Remote Control:");tv.setChannel(channel);}
}

🌟 Main函数展示桥接模式的使用

public class BridgePatternDemo {public static void main(String[] args) {// 创建不同品牌的电视TV sonyTV = new SonyTV();TV samsungTV = new SamsungTV();// 使用Sony Remote控制Sony TVRemoteControl sonyRemote = new SonyRemoteControl(sonyTV);sonyRemote.on();sonyRemote.setChannel(5);sonyRemote.off();System.out.println("-------------");// 使用Samsung Remote控制Samsung TVRemoteControl samsungRemote = new SamsungRemoteControl(samsungTV);samsungRemote.on();samsungRemote.setChannel(10);samsungRemote.off();}
}

运行结果:

Using Sony Remote Control:
Sony TV is now ON
Using Sony Remote Control:
Sony TV: Channel set to 5
Using Sony Remote Control:
Sony TV is now OFF
-------------
Using Samsung Remote Control:
Samsung TV is now ON
Using Samsung Remote Control:
Samsung TV: Channel set to 10
Using Samsung Remote Control:
Samsung TV is now OFF

源码应用

JDBC中的桥接模式源码探索:

背景:
在Java的数据库连接(JDBC)中,JDBC API 提供了统一的数据库操作接口,而具体的数据库操作实现则由各数据库厂商的驱动(Driver)实现。这里,JDBC API 是抽象部分,而每个数据库驱动则是具体实现部分。

桥接模式分析:

  • Abstraction(抽象部分): java.sql.DriverManager 提供了一个统一的接口,用于注册不同数据库的驱动。
  • Implementor(实现部分): 各个数据库驱动(例如MySQL、PostgreSQL、Oracle等),分别通过实现 java.sql.Driver 接口来具体执行数据库连接操作。

桥接效果:
程序员只需基于统一的JDBC接口操作数据库,而无需关心底层数据库具体如何实现连接。通过桥接模式,可以轻松扩展或替换不同的数据库驱动,而不影响客户端代码。

代码使用示例:

Connection connection = DriverManager.getConnection("jdbc:mysql://localhost:3306/mydb", "user", "password");

DriverManager 在内部会根据传入的 JDBC URL 查找合适的 Driver 来建立数据库连接,这样就将抽象的 Connection 操作与具体的数据库实现分离。

🌐 桥接模式的类图

+-------------------------------------+                 +--------------------------------+
|          java.sql.Driver            |  <---- Bridge   |  Abstract representation      |
|-------------------------------------|                 | of how to connect to database |
| + connect()                         |                 | using different implementations|
| + acceptURL()                       |                 +--------------------------------+
| + other common methods              |          
+----------------+--------------------+     ^|       Implementor|+-------------------------------+       +------------------------------------+|      com.mysql.cj.jdbc.Driver  |       |    org.postgresql.Driver           |+-------------------------------+       +------------------------------------+| + Implementation of connect()  |       |  + Implementation of connect()     |+-------------------------------+       +------------------------------------+^                                         ^|                                         |+-------------------------------+         ||      java.sql.DriverManager   |          |+-------------------------------+         ||  + registerDriver(Driver d)    |         ||  + getConnection(String url)   |         ||  + other common methods        |         |+-------------------------------+         ||+------------+-------------+|    java.sql.Connection   |+--------------------------+|  + createStatement()      ||  + close()                |+--------------------------+

⚙️ 关键源码解读:

1. Driver 接口(抽象部分,桥接接口)

package java.sql;import java.util.Properties;
import java.sql.SQLException;public interface Driver {// 尝试建立数据库连接Connection connect(String url, Properties info) throws SQLException;// 判断Driver是否可以处理传入的URLboolean acceptsURL(String url) throws SQLException;// 获取驱动的相关属性信息DriverPropertyInfo[] getPropertyInfo(String url, Properties info) throws SQLException;// 获取驱动的主版本号int getMajorVersion();// 获取驱动的次版本号int getMinorVersion();// 检查驱动是否为JDBC规范的合规实现boolean jdbcCompliant();
}

在这个接口中,Driver定义了通用的行为,所有数据库驱动都需要实现这些方法,如 connect()acceptsURL()

2. DriverManager(抽象部分与实现部分的桥梁)

package java.sql;import java.util.Enumeration;
import java.util.Vector;
import java.sql.Driver;
import java.sql.SQLException;public class DriverManager {// 注册驱动public static synchronized void registerDriver(java.sql.Driver driver) throws SQLException {registeredDrivers.addIfAbsent(new DriverInfo(driver));}// 通过指定URL连接数据库public static Connection getConnection(String url, String user, String password)throws SQLException {java.util.Properties info = new java.util.Properties();info.put("user", user);info.put("password", password);return getConnection(url, info);}// 具体的连接数据库逻辑public static Connection getConnection(String url, java.util.Properties info)throws SQLException {// 遍历已注册的驱动for (DriverInfo aDriver : registeredDrivers) {try {if (aDriver.driver.acceptsURL(url)) {return aDriver.driver.connect(url, info); // 调用驱动的connect方法}} catch (SQLException e) {// 忽略异常,尝试下一个驱动}}throw new SQLException("No suitable driver found for " + url);}// 保持已注册驱动的信息private static final CopyOnWriteArrayList<DriverInfo> registeredDrivers = new CopyOnWriteArrayList<>();static {// 内部静态代码块加载驱动loadInitialDrivers();}// 省略其他方法
}

解析:

  • DriverManager.registerDriver() 用于注册数据库驱动,开发者无需直接使用具体驱动类来注册。
  • DriverManager.getConnection() 是我们日常使用的API,内部通过调用已注册的 Driver 实现 connect() 方法来建立连接。

桥接模式应用: DriverManager 作为抽象的管理者,通过 Driver 接口的实现来具体连接到不同的数据库,真正实现了抽象和实现的分离。不同的数据库驱动(如MySQL、PostgreSQL等)是 Driver 接口的不同实现,开发者使用统一的 DriverManager.getConnection() 方法进行数据库操作,无需了解底层的实现。

3. com.mysql.cj.jdbc.Driver(具体实现部分)

package com.mysql.cj.jdbc;import java.sql.Driver;
import java.sql.Connection;
import java.sql.SQLException;
import java.util.Properties;public class Driver implements java.sql.Driver {static {try {// 在类加载时自动注册MySQL驱动java.sql.DriverManager.registerDriver(new Driver());} catch (SQLException e) {throw new RuntimeException("Failed to register MySQL driver.", e);}}@Overridepublic Connection connect(String url, Properties info) throws SQLException {// 检查是否支持该URLif (!acceptsURL(url)) {return null;}// 创建与MySQL数据库的连接return new ConnectionImpl(host, port, dbName, user, password);}@Overridepublic boolean acceptsURL(String url) {// 检查URL是否符合MySQL的格式return url.startsWith("jdbc:mysql:");}// 省略其他实现方法...
}

解析:

  • com.mysql.cj.jdbc.DriverDriver 接口的具体实现,通过桥接到 DriverManager,它实现了具体的连接MySQL数据库的逻辑。

  • acceptsURL() 方法判断传入的URL是否是MySQL的格式,connect() 方法则通过该URL与MySQL数据库建立连接。

  • JDBC桥接模式总结:

    • Abstraction(抽象部分): DriverManager 是抽象部分,提供统一的API供开发者使用,它只负责管理和协调不同的 Driver 实现,而不关心具体实现细节。
    • Implementor(实现部分): 各个数据库驱动实现了 Driver 接口,例如 com.mysql.cj.jdbc.Driverorg.postgresql.Driver,它们分别负责与具体的数据库进行交互。

    桥接模式通过分离抽象和实现,使得JDBC API可以轻松支持多种不同的数据库,而开发者只需通过统一的 DriverManager.getConnection() 进行操作。不同数据库的连接实现细节被隐藏在各自的 Driver 实现类中,确保了系统的扩展性和维护性。

📜 总结:

桥接模式让我们能够将抽象部分与具体实现分离,在保持系统灵活的同时,允许独立扩展抽象和实现层次。在上面的例子中,RemoteControlTV 分离,使得你可以轻松扩展更多不同类型的遥控器或电视品牌,而无需修改已有代码。

桥接模式在JDK和各种框架中得到广泛应用,特别是在处理跨平台差异、解耦抽象和实现的场景中。通过桥接模式,能够保持系统的高扩展性和灵活性,便于维护和改进。无论是数据库操作、GUI绘制、日志框架还是视图解析,桥接模式都扮演着核心角色。

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

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

相关文章

How to install OpenAI Universe without getting error code 1 on Windows?

题意&#xff1a;怎样在Windows上安装OpenAI Universe并避免出现错误代码1 问题背景&#xff1a; When I try to install OpenAi Universe on my Windows machine via python pip I get following stacktrace: 当我尝试通过 python pip 在我的 Windows 机器上安装 OpenAI Uni…

S-Procedure的基本形式及使用

理论 Lemma 1. ( S- Procedure[ 34] ) : Define the quadratic func- \textbf{Lemma 1. ( S- Procedure[ 34] ) : Define the quadratic func- } Lemma 1. ( S- Procedure[ 34] ) : Define the quadratic func- tions w.r.t. x ∈ C M 1 \mathbf{x}\in\mathbb{C}^M\times1 x…

dify的介绍

官方网址&#xff1a;https://dify.ai Dify 是一个开源的大型语言模型&#xff08;LLM&#xff09;应用开发平台&#xff0c;它提供了一个直观的用户界面&#xff0c;结合了AI工作流、检索-生成&#xff08;RAG&#xff09;管道、智能体&#xff08;Agent&#xff09;、模型管理…

Grafana 可视化配置

Grafana 是什么 Grafana 是一个开源的可视化和监控工具&#xff0c;广泛用于查看和分析来自各种数据源的时间序列数据。它提供了一个灵活的仪表盘&#xff08;dashboard&#xff09;界面&#xff0c;用户可以通过它将数据源中的指标进行图表化展示和监控&#xff0c;帮助分析趋…

国网B接口 USC安防平台 海康摄像机配置

国网B接口海康摄像机配置介绍 如下以海康DS-NACN6432I-GLN摄像机为例&#xff0c;配置国网B接口设备接入流程&#xff0c;海康摄像机的固件版本为 V5.6.11 build 210109 210107。该设备为球机&#xff0c;支持国网B接口云台控制功能。图标编号可以对应二者的配置。 注意 同一…

dify安装(传统docker-compose)

克隆 Dify 代码仓库 克隆 Dify 源代码至本地环境。 git clone --depth 1 https://github.com/langgenius/dify.git启动 Dify 进入 Dify 源代码的 Docker 目录 cd dify/docker-legacy启动 Docker 容器 docker-compose up -d运行命令后&#xff0c;你应该会看到类似以下的输…

Unity3D类似于桌面精灵的功能实现

前言&#xff1a; 由于最近在做游戏魔改&#xff0c;很多功能在游戏里面没法实现&#xff08;没错&#xff0c;说的就是排行榜&#xff09;&#xff0c;所以准备用Unity3D开发一个类似于桌面精灵的功能部件&#xff0c;实现效果如下&#xff1a; PS&#xff1a;有需要定制的老…

408算法题leetcode--第三天

1672. 最富有客户的资产总量 1672. 最富有客户的资产总量思路&#xff1a;双重循环遍历二维数组时间复杂度&#xff1a;O(mn)&#xff1b;空间&#xff1a;O(1) class Solution { public:int maximumWealth(vector<vector<int>>& accounts) {int ret 0;for(…

Java | Leetcode Java题解之第403题青蛙过河

题目&#xff1a; 题解&#xff1a; class Solution {public boolean canCross(int[] stones) {int n stones.length;boolean[][] dp new boolean[n][n];dp[0][0] true;for (int i 1; i < n; i) {if (stones[i] - stones[i - 1] > i) {return false;}}for (int i 1…

使用 Milvus、vLLM 和 Llama 3.1 搭建 RAG 应用

vLLM 是一个简单易用的 LLM 推理服务库。加州大学伯克利分校于 2024 年 7 月将 vLLM 作为孵化项目正式捐赠给 LF AI & Data Foundation 基金会。欢迎 vLLM 加入 LF AI & Data 大家庭&#xff01;&#x1f389; 在主流的 AI 应用架构中&#xff0c;大语言模型&#xff0…

Python数据处理利器,pivot与melt让表格变得灵活

大家好&#xff0c;在数据分析和处理过程中&#xff0c;数据的重塑是一个非常常见且重要的操作。数据重塑能够从不同的角度观察数据&#xff0c;以更符合分析需求的方式来呈现数据。在Python的Pandas库中&#xff0c;pivot和melt是两种强大的数据重塑工具&#xff0c;能够轻松地…

电离层闪烁

电离层闪烁&#xff0c;有的时候有有的时候无&#xff0c;但是经常出现&#xff0c;导致导航信号的振幅和相位发生变化&#xff0c;影响导航精度。使得载噪比降低。定位精度降低。 电离层闪烁的大小从几米到几百米&#xff0c;所以在使用RTK时&#xff0c;就算是相隔很近的基站…

npm 设置国内镜像源

1.1 镜像源概述 镜像源是软件包管理工具用来下载和安装软件包的服务器地址。由于网络原因&#xff0c;直接使用官方源可能会导致速度慢或连接失败的问题。国内镜像源可以提供更快的访问速度和更稳定的连接。 1.2 镜像源的选择 国内有许多可用的npm镜像源&#xff0c;包括但不限…

OKHttp实现原理分享

前言介绍 大约在2年半之前&#xff0c;就想写一篇关于OKHttp原理的文章&#xff0c;一来深入了解一下其原理&#xff0c;二来希望能在了解原理之后进行更好的使用。但是因为种种原因&#xff0c;一直无限往后推迟&#xff0c;最近因为我们情景智能半个月一次的分享轮到我了&…

【鸿蒙】HarmonyOS NEXT星河入门到实战1-开发环境准备

目录 一、达成目标 二、鸿蒙开发环境准备 2.1 开发者工作下载 2.2 解压安装 2.3 运行配置安装node.js和SDK 2.4 开始创建第一个项目 2.5 预览 2.5.1 预览遇到的问题&#xff08;报错&#xff09; 2.5.2 修改内容查看预览 三、备用下载地址&#xff08;如果下载是4.X版…

Prism库:详解其核心组件和使用方法

Prism库简介 Prism库是一个开源项目&#xff0c;由 Microsoft 社区开发和维护。它是一组用于创建 WPF、UWP 和 Xamarin 应用程序的工具和库&#xff0c;提供了一种基于模块化和依赖注入的架构模式&#xff0c;同时它提供了一系列的工具&#xff0c;帮助开发人员构建可扩展、可…

MATLAB、FPGA、STM32中调用FFT计算频率、幅值及相位差

系列文章目录 文章目录 系列文章目录前言MATLABSTM32调用DSPSTM32中实现FFT关于初相位 FPGA 前言 最近在学习如何在STM32中调用FFT MATLAB 首先对FFT进行一下说明&#xff0c;我们输入N个点的数据到FFT中&#xff0c;FFT会返回N个点的数据&#xff0c;这些数据都是复数&#…

ASP.NET Core 入门教学二十八 linux打包部署

在Linux上打包和部署ASP.NET Core应用程序涉及几个步骤。以下是一个详细的指南&#xff0c;帮助你在Linux系统上完成这一过程。 1. 准备工作 确保你的Linux系统已经安装了以下软件&#xff1a; .NET SDK&#xff08;用于构建应用程序&#xff09;.NET Runtime&#xff08;用…

ctfshow-PHP反序列化

web254 源码 <?php/* # -*- coding: utf-8 -*- # Author: h1xa # Date: 2020-12-02 17:44:47 # Last Modified by: h1xa # Last Modified time: 2020-12-02 19:29:02 # email: h1xactfer.com # link: https://ctfer.com //mytime 2023-12-4 0:22 */ error_reporting(0)…

Charles mac电脑配置

安装 Charles&#xff1a; 如果你还没有安装 Charles&#xff0c;可以从官方网站下载安装包并按照提示完成安装。 启动 Charles&#xff1a; 安装完成后&#xff0c;启动 Charles 应用程序。 设置 Charles 代理&#xff1a; Charles 默认的代理端口是 8888。你可以通过以下步…