Java桥接模式源码剖析及使用场景
目录
- 一、介绍
- 二、项目管理系统中使用桥接模式
- 三、权限管理中使用桥接模式
- 四、Java JDBC中使用桥接模式
一、介绍
它的主要目的是将抽象化与实现化分离,使得二者可以独立变化,就像一个桥,将两个变化维度连接起来。各个维度都可以独立的变化。故称之为:桥接模式
桥接模式的核心在于通过一个桥接接口,将抽象部分与实现部分解耦。这样做的好处是显而易见的,当系统中的某个维度(抽象或实现)需要变更时,不会影响到另一个维度。具体来说,桥接模式涉及以下几个关键角色:
-
抽象化角色(Abstraction):它是抽象类的接口,定义了一个实现化的引用和对实现化的操作。
-
修正抽象化角色(Refined Abstraction):它是扩展了抽象化角色的子类,在实际应用中,具体的业务逻辑通常在这个角色中实现。
-
实现化角色(Implementor):这个角色定义了一个接口,由具体实现化角色来实现。
-
具体实现化角色(Concrete Implementor):这个角色实现了实现化接口的具体类。
二、项目管理系统中使用桥接模式
需求:项目管理系统,包含多种任务(编码任务,测试任务)和多种任务执行(本地执行,远程执行)方式 。还有开发和测试角色。我们可以使用桥接模式来分离任务和任务执行方式的实现,以便它们可以独立变化。
- 定义项目管理系统中的任务接口
public interface Task {void performTask();
}
- 具体的任务实现类
// 编码任务
public class CodingTask implements Task {@Overridepublic void performTask() {System.out.println("Performing coding task");}
}// 测试任务
public class TestingTask implements Task {@Overridepublic void performTask() {System.out.println("Performing testing task");}
}
- 任务执行方式接口
public interface TaskExecution {void executeTask();
}
- 具体的任务执行方式实现类
// 本地执行任务
public class LocalTaskExecution implements TaskExecution {@Overridepublic void executeTask() {System.out.println("Executing task locally");}
}// 远程执行任务
public class RemoteTaskExecution implements TaskExecution {@Overridepublic void executeTask() {System.out.println("Executing task remotely");}
- 项目管理系统角色抽象类
public abstract class ProjectRole {protected Task task;protected TaskExecution taskExecution;public void setTask(Task task) {this.task = task;}public void setTaskExecution(TaskExecution taskExecution) {this.taskExecution = taskExecution;}public abstract void performProjectTask();
}
- 项目管理系统具体角色实现类
// 开发人员角色
public class Developer extends ProjectRole {@Overridepublic void performProjectTask() {System.out.print("Developer role: ");task.performTask();System.out.print("Using ");taskExecution.executeTask();}
}// 测试人员角色
public class Tester extends ProjectRole {@Overridepublic void performProjectTask() {System.out.print("Tester role: ");task.performTask();System.out.print("Using ");taskExecution.executeTask();}
}
- 客户端代码演示桥接模式的使用
public class BridgePatternDemo {public static void main(String[] args) {Task codingTask = new CodingTask();Task testingTask = new TestingTask();TaskExecution localTaskExecution = new LocalTaskExecution();TaskExecution remoteTaskExecution = new RemoteTaskExecution();// 创建不同角色的实例ProjectRole developer = new Developer();ProjectRole tester = new Tester();// 设置不同的任务和任务执行方式developer.setTask(codingTask);developer.setTaskExecution(remoteTaskExecution);tester.setTask(testingTask);tester.setTaskExecution(localTaskExecution);// 视图数据developer.performProjectTask(); // Output: "Developer role: Performing coding task Using Executing task remotely"tester.performProjectTask(); // Output: "Tester role: Performing testing task Using Executing task locally"}
}
大家说上面七步是桥接模式吗?在实际项目中,我们可以根据需要添加更多的任务和任务执行方式,而不需要修改已有的代码。核心在于正确识别并分离出两个独立的维度,并且这两个维度应该能够各自独立地变化和扩展。
三、权限管理中使用桥接模式
需求:权限管理系统,分别有普通角色、管理员角色、超级管理员角色。以数据权限设计,普通角色只能看到自己操作的数据,管理员角色可以看到普通角色操作的数据,超级管理员角色可以看到所有角色的数据。
- 定义数据权限控制策略接口
public interface DataPermissionControlStrategy {void process();
}
- 定义具体的数据权限控制策略实现类
// 普通角色只能看到自己操作的数据
public class NormalDataPermissionControlStrategy implements DataPermissionControlStrategy {@Overridepublic void process() {System.out.println("Normal role can only access own data");}
}// 管理员角色可以看到普通角色操作的数据
public class AdminDataPermissionControlStrategy implements DataPermissionControlStrategy {@Overridepublic void process() {System.out.println("Admin role can access normal role's data");}
}// 超级管理员角色可以看到所有角色的数据
public class SuperAdminDataPermissionControlStrategy implements DataPermissionControlStrategy {@Overridepublic void process() {System.out.println("Super admin role can access all data");}
}
- 定义用户角色抽象类
public abstract class UserRole {protected DataPermissionControlStrategy dataPermissionControlStrategy;public void setDataPermissionControlStrategy(DataPermissionControlStrategy dataPermissionControlStrategy) {this.dataPermissionControlStrategy = dataPermissionControlStrategy;}public abstract void viewData();
}
- 定义具体的用户角色实现类
// 普通角色
public class NormalRole extends UserRole {@Overridepublic void viewData() {System.out.print("Normal role: ");dataPermissionControlStrategy.process();}
}// 管理员角色
public class AdminRole extends UserRole {@Overridepublic void viewData() {System.out.print("Admin role: ");dataPermissionControlStrategy.process();}
}// 超级管理员角色
public class SuperAdminRole extends UserRole {@Overridepublic void viewData() {System.out.print("Super admin role: ");dataPermissionControlStrategy.process();}
}
- 客户端代码演示桥接模式的使用
// 客户端代码演示桥接模式的使用
public class BridgePatternDemo {public static void main(String[] args) {DataPermissionControlStrategy normalStrategy = new NormalDataPermissionControlStrategy();DataPermissionControlStrategy adminStrategy = new AdminDataPermissionControlStrategy();DataPermissionControlStrategy superAdminStrategy = new SuperAdminDataPermissionControlStrategy();// 创建不同角色的实例UserRole normalUser = new NormalRole();UserRole adminUser = new AdminRole();UserRole superAdminUser = new SuperAdminRole();// 设置不同的数据权限控制策略normalUser.setDataPermissionControlStrategy(normalStrategy);adminUser.setDataPermissionControlStrategy(adminStrategy);superAdminUser.setDataPermissionControlStrategy(superAdminStrategy);// 视图数据normalUser.viewData(); // Output: "Normal role: Normal role can only access own data"adminUser.viewData(); // Output: "Admin role: Admin role can access normal role's data"superAdminUser.viewData(); // Output: "Super admin role: Super admin role can access all data"}
}
四、Java JDBC中使用桥接模式
在Java JDBC中,桥接模式并不是直接使用的设计模式。然而,在JDBC中,可以看到一些类似于桥接模式的结构和思想。
1.DataSource 接口:在JDBC中,DataSource 是一个接口,它提供了获取数据库连接的方法。不同的数据库厂商(比如MySQL、Oracle等)都会有对应的DataSource 实现类。这种设计类似于桥接模式中的抽象部分。
2.Connection 接口和实现类:Connection 接口代表着一个数据库连接,而具体的数据库连接操作是由各个数据库厂商提供的实现类来完成的,比如MySQLConnection、OracleConnection 等。这也体现了桥接模式中的实现部分。
通过接口和实现类的结合,实现了对不同数据库的访问和操作,并且很好地解耦了抽象和实现部分。
// 1.定义厂商DataSource 接口
public interface DataSource {Connection getConnection();
}// 2.MySQLDataSource 实现类
public class MySQLDataSource implements DataSource {@Overridepublic Connection getConnection() {// 返回 MySQL 数据库连接return new MySQLConnection();}
}// 3.OracleDataSource 实现类
public class OracleDataSource implements DataSource {@Overridepublic Connection getConnection() {// 返回 Oracle 数据库连接return new OracleConnection();}
}// 4.不同的数据库不同的连接方法Connection 接口
public interface Connection {void executeQuery(String query);
}// 5.MySQLConnection 实现类
public class MySQLConnection implements Connection {@Overridepublic void executeQuery(String query) {// 在 MySQL 数据库中执行查询System.out.println("Executing query in MySQL database: " + query);}
}// 6.OracleConnection 实现类
public class OracleConnection implements Connection {@Overridepublic void executeQuery(String query) {// 在 Oracle 数据库中执行查询System.out.println("Executing query in Oracle database: " + query);}
}相关文章:
Java桥接模式源码剖析及使用场景
目录 一、介绍二、项目管理系统中使用桥接模式三、权限管理中使用桥接模式四、Java JDBC中使用桥接模式 一、介绍 它的主要目的是将抽象化与实现化分离,使得二者可以独立变化,就像一个桥,将两个变化维度连接起来。各个维度都可以独立的变化。…...
【异常处理】verilator安装时出现异常 make: *** [Makefile:195: verilator_gantt.1] Error 13
在ubuntu中安装verilator工具时执行make出现该报错。 当我出现这个报错的时候我一脸懵逼,因为网上找不到相关解决办法。 后来想到我的verilator是从github上下载zip,然后解压后传到ubuntu上的,windows上解压我记得会把-替换成_,这…...
测试一下 Anthropic 宣称超过 GPT-4 的 Claude 3 Opus
测试一下 Anthropic 宣称超过 GPT-4 的 Claude 3 Opus 0. 引言1. 测试 Claude 3 Opus3. 试用 api key 限制 0. 引言 今天测试一下 Anthropic 发布的 Claude 3 Opus。 3月4日,Anthropic 宣布推出 Claude 3 型号系列,该系列在广泛的认知任务中树立了新的…...
【题解】—— LeetCode一周小结10
【题解】—— 每日一道题目栏 上接:【题解】—— LeetCode一周小结9 4.用栈实现队列 题目链接:232. 用栈实现队列 请你仅使用两个栈实现先入先出队列。队列应当支持一般队列支持的所有操作(push、pop、peek、empty):…...
Android studio虚拟调试出现“我的APP keeps stopping”问题
问题如图: 遇到这种情况,一看代码,也没有报错呀,怎么不能运行呢?不要慌!我们一步一步来。 1、查看Logcat日志 在Android Studio中查看Logcat窗口,可以获取应用程序崩溃时的详细错误信息&…...
【Web】浅聊Java反序列化之Spring2链——两层动态代理
目录 简介 简话JdkDynamicAopProxy 关于target的出身——AdvisedSupport EXP 请确保已阅读过前文或对Spring1链至少有一定认知:【Web】浅聊Java反序列化之Spring1链——三层动态代理-CSDN博客 简介 Spring2 和 Spring1 的反序列化过程基本相同,唯一…...
2386. 找出数组的第 K 大和
2386. 找出数组的第 K 大和 题目链接:2386. 找出数组的第 K 大和 代码如下: //优先队列 //参考:https://leetcode.cn/problems/find-the-k-sum-of-an-array/solutions/2668280/zhao-chu-shu-zu-de-di-k-da-he-by-leetcod-z5kq class Soluti…...
Pytorch学习 day10(L1Loss、MSELoss、交叉熵Loss)
Loss loss的作用如下: 计算实际输出和真实值之间的差距为我们更新模型提供一定的依据(反向传播) L1Loss 绝对值损失函数:在每一个batch_size内,求每个输入x和标签y的差的绝对值,最后返回他们平均值 M…...
2.2 传统经济学在耍赖
传统经济学中,主体的行为决策是研究的重点对幸福的追求不是传统经济学的研究重点,决策才是。在传统经济学那里,只要能搞清楚是什么决定了决策就可以了。 传统经济学用人们对物品的喜好的排序去替代了对幸福的直接度量。这样做有一个好处&…...
【算法面试题】-04
执行时长 def min_execution_time(n, size, tasks):a 0ans sizei 0while i < size:tmp tasks[i]a tmpif a < n:a 0else:a - ni 1ans a // nif a % n ! 0:ans 1return ans# 读取输入 n int(input()) size int(input()) tasks list(map(int, input().split()))…...
力扣hot100:152.乘积最大子数组(动态规划)
一个子数组问题,我们要使用线性dp,最好先考虑以i结尾,如果定义dp[i]为前i个数最大子数组乘积值 那么dp[i-1]就无法转移到dp[i]。因此我们先考虑dp[i]定义为以第i个数结尾的最大子数组乘积值。 53. 最大子数组和 最大子数组和是一个动态规划问…...
【python 】----Pytest基础知识与进阶知识
定义 用于编写和执行Python测试全功能测试框架(工具),是一个第三方库 安装 pip insatll pytest 安装pytest --version 校验 pytest的组成构成 不写调用语句也可以执行函数内容 在用例运行语句里面: -s:指的是开启与终端的交互,如果没有-s(程序不会输入与打印),一条用…...
谷歌开源的LLM大模型 Gemma 简介
相关链接: Hugging face模型下载地址:https://huggingface.co/google/gemma-7bGithub地址:https://github.com/google/gemma_pytorch论文地址:https://storage.googleapis.com/deepmind-media/gemma/gemma-report.pdf官方博客&…...
深入理解 Vuex:从基础到应用场景
前言 在之前的文章中,我们已经对 Vue.js 有了一定的了解。今天我们要对Vue官方的状态共享管理器Vuex进行详细讲解,将其基本吃透,目标是面对大多数业务需求; 一、介绍 Vuex 是一个专为 Vue.js 应用程序开发的状态管理模式。它采用…...
自定义 classNames hooks
什么是自定义 hooks 自定义hooks是react提供的编写公共函数的方法 自定hooks 和 通用函数的区别 一定有人会说 hooks 可以使用react 的方法,但是公共函数也可以,因为 jsx 语法的原因 函数必须开头进行大写 其实这些都是 react 的语法规范ÿ…...
玩转centos 下的core 文件
玩转centos 下的core 文件 ------------------------------------------------------------ author: hjjdebug date: 2024年 03月 06日 星期三 12:38:35 CST description: 玩转centos 下的core 文件 ------------------------------------------------------------ 一: 准备一…...
深入浅出计算机网络 day.1 概论③ 电路交换、分组交换和报文交换
人无法同时拥有青春和对青春的感受 —— 04.3.9 内容概述 01.电路交换、分组交换和报文交换 02.三种交换方式的对比 一、电路交换、分组交换和报文交换 1.电路交换 计算机之间的数据传送是突发式的,当使用电路交换来传送计算机数据时,其线路的传输效率一…...
linux:线程的控制
个人主页 : 个人主页 个人专栏 : 《数据结构》 《C语言》《C》《Linux》 文章目录 前言一、线程的总结1. 线程的优点2. 线程的缺点3. 线程异常4.线程和进程 二、线程的控制创建线程线程终止线程等待获取返回值 线程分离 总结 前言 本文作为我对于线程的…...
小程序分账方案:实现商户分账的简便与灵活
随着移动支付的普及和小程序的快速发展,越来越多的商家选择在微信小程序上开展业务。然而,对于一些有多个分账方的商户而言,如何实现快速、准确和灵活的资金分账成为了一个挑战。本文将介绍一种高效的小程序分账方案,帮助商户轻松…...
Python数值微积分,摆脱被高数支配的恐惧
文章目录 差分和累加积分多重积分 Python科学计算:数组💯数据生成 差分和累加 微积分是现代科学最基础的数学工具,但其应用对象往往是连续函数,而其在非连续函数的类比,便是差分与累加。在【numpy】中,可…...
Python|GIF 解析与构建(5):手搓截屏和帧率控制
目录 Python|GIF 解析与构建(5):手搓截屏和帧率控制 一、引言 二、技术实现:手搓截屏模块 2.1 核心原理 2.2 代码解析:ScreenshotData类 2.2.1 截图函数:capture_screen 三、技术实现&…...
Linux应用开发之网络套接字编程(实例篇)
服务端与客户端单连接 服务端代码 #include <sys/socket.h> #include <sys/types.h> #include <netinet/in.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <arpa/inet.h> #include <pthread.h> …...
渗透实战PortSwigger靶场-XSS Lab 14:大多数标签和属性被阻止
<script>标签被拦截 我们需要把全部可用的 tag 和 event 进行暴力破解 XSS cheat sheet: https://portswigger.net/web-security/cross-site-scripting/cheat-sheet 通过爆破发现body可以用 再把全部 events 放进去爆破 这些 event 全部可用 <body onres…...
测试markdown--肇兴
day1: 1、去程:7:04 --11:32高铁 高铁右转上售票大厅2楼,穿过候车厅下一楼,上大巴车 ¥10/人 **2、到达:**12点多到达寨子,买门票,美团/抖音:¥78人 3、中饭&a…...
对WWDC 2025 Keynote 内容的预测
借助我们以往对苹果公司发展路径的深入研究经验,以及大语言模型的分析能力,我们系统梳理了多年来苹果 WWDC 主题演讲的规律。在 WWDC 2025 即将揭幕之际,我们让 ChatGPT 对今年的 Keynote 内容进行了一个初步预测,聊作存档。等到明…...
从零开始打造 OpenSTLinux 6.6 Yocto 系统(基于STM32CubeMX)(九)
设备树移植 和uboot设备树修改的内容同步到kernel将设备树stm32mp157d-stm32mp157daa1-mx.dts复制到内核源码目录下 源码修改及编译 修改arch/arm/boot/dts/st/Makefile,新增设备树编译 stm32mp157f-ev1-m4-examples.dtb \stm32mp157d-stm32mp157daa1-mx.dtb修改…...
QT: `long long` 类型转换为 `QString` 2025.6.5
在 Qt 中,将 long long 类型转换为 QString 可以通过以下两种常用方法实现: 方法 1:使用 QString::number() 直接调用 QString 的静态方法 number(),将数值转换为字符串: long long value 1234567890123456789LL; …...
安卓基础(aar)
重新设置java21的环境,临时设置 $env:JAVA_HOME "D:\Android Studio\jbr" 查看当前环境变量 JAVA_HOME 的值 echo $env:JAVA_HOME 构建ARR文件 ./gradlew :private-lib:assembleRelease 目录是这样的: MyApp/ ├── app/ …...
【JVM面试篇】高频八股汇总——类加载和类加载器
目录 1. 讲一下类加载过程? 2. Java创建对象的过程? 3. 对象的生命周期? 4. 类加载器有哪些? 5. 双亲委派模型的作用(好处)? 6. 讲一下类的加载和双亲委派原则? 7. 双亲委派模…...
从 GreenPlum 到镜舟数据库:杭银消费金融湖仓一体转型实践
作者:吴岐诗,杭银消费金融大数据应用开发工程师 本文整理自杭银消费金融大数据应用开发工程师在StarRocks Summit Asia 2024的分享 引言:融合数据湖与数仓的创新之路 在数字金融时代,数据已成为金融机构的核心竞争力。杭银消费金…...
