【Java设计模式】Bridge模式:在Java中解耦抽象与实现

news2025/1/2 0:30:40

文章目录

  • 【Java设计模式】Bridge模式:在Java中解耦抽象与实现
    • 一、概述
    • 二、Bridge设计模式的别名
    • 三、Bridge设计模式的意图
    • 四、Bridge模式的详细解释及实际示例
    • 五、Java中Bridge模式的编程示例
    • 六、Bridge模式类图
    • 七、Java中何时使用Bridge模式
    • 八、Java中Bridge模式的实际应用
    • 九、Bridge模式的优点和权衡
    • 十、源码下载

【Java设计模式】Bridge模式:在Java中解耦抽象与实现

一、概述

在Java开发中,设计模式是提高代码质量和可维护性的重要工具。本文将介绍Bridge模式,它是一种结构型设计模式,用于解耦抽象与实现,增强软件系统的灵活性和可扩展性。

二、Bridge设计模式的别名

  • Handle/Body

三、Bridge设计模式的意图

Bridge设计模式是Java中的一种结构模式,它将抽象与其实现解耦,允许两者独立变化。该模式对于开发灵活和可扩展的软件系统至关重要。

四、Bridge模式的详细解释及实际示例

  1. 实际示例
    • 在Java中,Bridge模式常用于GUI框架、数据库驱动程序和设备驱动程序中。例如,一个通用遥控器(抽象)可以通过一致的接口操作各种电视品牌(实现)。
    • 想象一个通用遥控器(抽象)可以操作不同品牌和类型的电视(实现)。遥控器提供了一致的接口来执行诸如打开/关闭、更改频道和调整音量等操作。每个电视品牌或类型都有自己特定的这些操作的实现。通过使用Bridge模式,遥控器接口与电视实现解耦,允许遥控器与任何电视一起工作,无论其品牌或内部工作方式如何。这种分离允许添加新的电视型号而无需更改遥控器的代码,并且可以开发不同的遥控器来与同一组电视一起工作。
  2. 通俗解释
    • Bridge模式是关于优先使用组合而不是继承。实现细节从一个层次结构推到另一个具有单独层次结构的对象。
  3. 维基百科解释
    • Bridge模式是一种用于软件工程的设计模式,旨在“将抽象与其实现解耦,以便两者可以独立变化”。

五、Java中Bridge模式的编程示例

想象一下,你有一件武器可以有各种附魔,你需要将不同的武器与不同的附魔结合起来。你会如何处理这个问题?你会为每个武器创建多个副本,每个副本都有不同的附魔,还是会创建单独的附魔并根据需要将它们应用于武器?Bridge模式使你能够做到后者。

这里我们有Weapon层次结构:

public interface Weapon {
    void wield();
    void swing();
    void unwield();
    Enchantment getEnchantment();
}
public class Sword implements Weapon {
    private final Enchantment enchantment;
    public Sword(Enchantment enchantment) {
        this.enchantment = enchantment;
    }
    @Override
    public void wield() {
        LOGGER.info("The sword is wielded.");
        enchantment.onActivate();
    }
    @Override
    public void swing() {
        LOGGER.info("The sword is swung.");
        enchantment.apply();
    }
    @Override
    public void unwield() {
        LOGGER.info("The sword is unwielded.");
        enchantment.onDeactivate();
    }
    @Override
    public Enchantment getEnchantment() {
        return enchantment;
    }
}
public class Hammer implements Weapon {
    private final Enchantment enchantment;
    public Hammer(Enchantment enchantment) {
        this.enchantment = enchantment;
    }
    @Override
    public void wield() {
        LOGGER.info("The hammer is wielded.");
        enchantment.onActivate();
    }
    @Override
    public void swing() {
        LOGGER.info("The hammer is swung.");
        enchantment.apply();
    }
    @Override
    public void unwield() {
        LOGGER.info("The hammer is unwielded.");
        enchantment.onDeactivate();
    }
    @Override
    public Enchantment getEnchantment() {
        return enchantment;
    }
}

这里是单独的Enchantment层次结构:

public interface Enchantment {
    void onActivate();
    void apply();
    void onDeactivate();
}
public class FlyingEnchantment implements Enchantment {
    @Override
    public void onActivate() {
        LOGGER.info("The item begins to glow faintly.");
    }
    @Override
    public void apply() {
        LOGGER.info("The item flies and strikes the enemies finally returning to owner's hand.");
    }
    @Override
    public void onDeactivate() {
        LOGGER.info("The item's glow fades.");
    }
}
public class SoulEatingEnchantment implements Enchantment {
    @Override
    public void onActivate() {
        LOGGER.info("The item spreads bloodlust.");
    }
    @Override
    public void apply() {
        LOGGER.info("The item eats the soul of enemies.");
    }
    @Override
    public void onDeactivate() {
        LOGGER.info("Bloodlust slowly disappears.");
    }
}

下面是两个层次结构的实际应用:

public static void main(String[] args) {
    LOGGER.info("The knight receives an enchanted sword.");
    var enchantedSword = new Sword(new SoulEatingEnchantment());
    enchantedSword.wield();
    enchantedSword.swing();
    enchantedSword.unwield();
    LOGGER.info("The valkyrie receives an enchanted hammer.");
    var hammer = new Hammer(new FlyingEnchantment());
    hammer.wield();
    hammer.swing();
    hammer.unwield();
}

以下是控制台输出。

The knight receives an enchanted sword.
The sword is wielded.
The item spreads bloodlust.
The sword is swung.
The item eats the soul of enemies.
The sword is unwielded.
Bloodlust slowly disappears.
The valkyrie receives an enchanted hammer.
The hammer is wielded.
The item begins to glow faintly.
The hammer is swung.
The item flies and strikes the enemies finally returning to owner's hand.
The hammer is unwielded.
The item's glow fades.

六、Bridge模式类图

在这里插入图片描述

七、Java中何时使用Bridge模式

考虑在以下情况下使用Bridge模式:

  1. 当你需要避免抽象与其实现之间的永久绑定,例如在运行时必须选择或切换实现时。
  2. 抽象和它们的实现都应该通过子类化进行扩展,允许每个组件独立扩展。
  3. 对抽象实现的更改不应影响客户端,意味着他们的代码不应需要重新编译。
  4. 当你的层次结构中出现大量类时,表明需要将一个对象拆分为两个部分,Rumbaugh将此概念称为“嵌套泛化”。
  5. 当你想在多个对象之间共享一个实现,可能使用引用计数,同时将此细节隐藏在客户端之外,如Coplien的String类所示,其中多个对象可以共享相同的字符串表示。

八、Java中Bridge模式的实际应用

  1. GUI框架,其中抽象是窗口,实现可以是底层的操作系统窗口系统。
  2. 数据库驱动程序,其中抽象是通用数据库接口,实现是特定数据库的驱动程序。
  3. 设备驱动程序,其中抽象是设备无关代码,实现是设备相关代码。

九、Bridge模式的优点和权衡

  1. 优点
    • 解耦接口和实现:Bridge模式通过将接口(高级操作)与实现(低级操作)分离来增强模块化。
    • 提高可扩展性:你可以独立扩展抽象和实现层次结构。
    • 隐藏实现细节:客户端只看到抽象的接口,而不是其实现。
  2. 权衡
    • 增加复杂性:该模式可能会使系统架构和代码复杂化,特别是对于不熟悉该模式的客户端。
    • 运行时开销:额外的抽象层可能会引入性能损失,尽管在实践中通常可以忽略不计。

十、源码下载

Bridge模式示例代码下载

通过本文的介绍,相信大家对Java中的Bridge模式有了更深入的了解。在实际开发中,合理运用Bridge模式可以提高代码的灵活性和可扩展性,同时降低代码的耦合性,为构建高质量的软件系统提供有力支持。

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

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

相关文章

如何使用ssm实现毕业生交流学习平台+vue

TOC ssm306毕业生交流学习平台vue 绪论 1.1 研究背景 现在大家正处于互联网加的时代,这个时代它就是一个信息内容无比丰富,信息处理与管理变得越加高效的网络化的时代,这个时代让大家的生活不仅变得更加地便利化,也让时间变得…

某系统任意用户创建漏洞

初来人间不知苦,潦倒半生一身无,转身回望来时路,方知生时为何哭。 漏洞描述 某系统存在任意用户创建漏洞,发送特定的请求包攻击者可以创建管理员账户登录后台 漏洞实战 出现漏洞的文件为 userproce.php,出现漏洞的…

汽车免拆诊断案例 | 马自达CX-3无音频输出

故障现象: 使用触摸屏打开收音机时,单选按钮打开收音机,但无法访问菜单。使用中控台中的旋转控制旋钮时,也会遇到相同的情况。 没有音频输出到车上的任何扬声器,包括卫星导航、蓝牙或语音识别。音量调节也不起作用&a…

食家巷大烤馍,美味超实在

在美食的世界里,总有一些味道能勾起我们内心深处的温暖回忆,食家巷大烤馍便是其中之一。 每一个食家巷大烤馍都是匠心的结晶。首先,精选优质的面粉,确保了烤馍的基础品质。这些面粉经过严格的筛选,颗粒细腻&#xff0c…

2024/8/28 英语每日一段

“Its great were getting these investments, but you actually cant materialize the benefits of these investments if the cities actually cant afford to run it,” he said. All of the eight cities say the planned expansions, such as new light rail lines in Ott…

掌握高效管理技巧:9款顶级待办事项软件

本文将介绍9款优质待办事项清单工具:1.PingCode;2.Worktile;3.滴答清单 (TickTick);4.日事清;5.效能工作任务;6.印象笔记;7.MyLifeOrganized;8.Quire;9.OmniFocus。 在当…

CTFhub通关攻略-SSRF篇【1-5关】

01关 内网访问 根据题意,它让我们去尝试访问127.0.0.1的flag.php,我们点进题目链接 有一个url参数可以进行输入,我们直接访问127.0.0.1的flag.php 这样就得到了flag 02 伪协议读取文件 点开题目链接发现有一个url的参数可以进行填写 题中说让…

【贪心 决策包容性 】757. 设置交集大小至少为2

本文涉及知识点 贪心 决策包容性 LeetCode757. 设置交集大小至少为2 给你一个二维整数数组 intervals ,其中 intervals[i] [starti, endi] 表示从 starti 到 endi 的所有整数,包括 starti 和 endi 。 包含集合 是一个名为 nums 的数组,并…

redis核心数据结构源码分析

dictEntry和redisObject 在 Redis 的实现中,当一个键值对被创建并存储时,键通常是一个字符串,而值则是一个 redisObject。因此,在 dictEntry 结构中,key 成员指向的是一个字符串,而 v.val 成员则指向一个 …

《深入理解JAVA虚拟机(第2版)》- 第2章 - 学习笔记

第二章 Java内存区域与内存溢出异常 2.1 概述 JVM是自动内存管理 2.2 运行时数据区 所谓运行时数据区是JVM在运行Java程序的时候将所管理的内存划分为几块不同的数据区域,分为:程序计数器、Java虚拟机栈、本地方法栈、Java堆、方法区,如下…

樊振东代言LV旗下品牌,媒介易解读冠军代言背后的商业逻辑

​近日,法国奢侈品巨头LVMH旗下的德国知名拉杆箱品牌RIMOWA宣布,中国乒乓球奥运冠军樊振东成为其品牌挚友。这一合作不仅是RIMOWA对樊振东在巴黎奥运会上出色表现的认可,也标志着品牌与体育冠军联手打造品牌形象的趋势日益显著。 樊振东&…

Moco求解最优化问题使用教程

理论部分 最简单的例子,流程 输出结果分析 理论部分 moco最终是调用CasAdi求解器来进行求解 对不常见的几个符号表达式含义进行解释: 多刚体动力学公式代表系统中,f_inertial (惯性力和科里奥利力);f_app (外力和接触力)&…

SQL注入-ctfshow

首先还是对sql的具体分析和讲解 原理: SQL注入是一种安全漏洞,它允许攻击者通过在应用程序的输入中插入或者操作SQL命令来改变后端数据库的查询和操作。SQL注入的主要原因是代码中直接将用户输入与SQL命令拼接在一起,没有进行适当的验证或清…

网络安全可以从事哪些岗位岗位职责是什么网络安全专业的就业前景

网络安全可以从事哪些岗位 伴随着社会的发展,网络安全被列为国家安全战略的一部分,因此越来越多的行业开始迫切需要网安人员,也有不少人转行学习网络安全。那么网络安全可以从事哪些岗位?岗位职责是什么?相信很多人都不太了解,…

『功能项目』着色器光透魔法球Shaders【09】

我们打开上一篇08技能释放的项目, 本章要做的事情是为魔法球增添一个光透效果shaders。 首先在Assets中创建一个Shaders文件夹 接着将场景中的灯光调暗一些(避免灯光太强压过将要设置半透明光透效果的魔法球) 将新Resources中的Shpere拖拽至…

MQTT服务器-mosquitto配置

我们要使用ESP8266使得STM32能够和服务器之间传递数据,需要有一台MQTT服务器。当然读者可以使用腾讯云、阿里云、OneNet等平台提供的MQTT服务更方便一些。 逻辑是这样的:我们首先需要一台服务器A作为中转站,然后我们的STM32作为客户端能够发…

Java接口中的长连接与短连接详解:概念、应用场景及实现

个人名片 🎓作者简介:java领域优质创作者 🌐个人主页:码农阿豪 📞工作室:新空间代码工作室(提供各种软件服务) 💌个人邮箱:[2435024119qq.com] &#x1f4f1…

2024年十款好用的图纸加密软件推荐|有效的图纸加密方法分享

在数字化时代,图纸作为企业的重要资产,其安全性显得尤为重要。随着技术的不断进步,市场上涌现出多款优秀的图纸加密软件,为企业和个人提供了强有力的安全保障。本文将为您推荐2024年十款好用的图纸加密软件。 1.安秉图纸加密软件…

JavaSE 面试题 46-50

使用runnable需重写run方法,而且返回值为viod型,等于没有返回值; Thread 类在调用 start()函数后就是执行的 是 Runnable 的 run()函数。 callable需重写call方法,call方法可以有返回值,支持泛型而且可以捕获解决异常&…

秋招突击——知识复习——HTTP/2、HTTP/3的改良

文章目录 引言正文HTTP/1.1与HTTP/1.01、长连接代替短链接2、管道传输缺点 HTTP2.0和HTTP1.11、头部压缩2、二进制格式3、并发传输4、服务器主动推送资源缺点 HTTP/3和HTTP/21、无队头阻塞2、更快的连接建立3、连接迁移 面试题1、HTTP是长连接还是短链接?2、HTTP长连…