设计模式之桥接模式:抽象与实现之间的分离艺术

在这里插入图片描述

~犬📰余~

“我欲贱而贵,愚而智,贫而富,可乎?
曰:其唯学乎”

桥接模式概述与角色组成

想象一下你家里的电视遥控器,无论是索尼还是三星的电视机,遥控器的按键功能都差不多:有开关按钮、音量调节、频道切换等。遥控器本身的功能(抽象)和具体品牌电视的实现(实现)是分离的,这就是桥接模式的一个生动写照。
桥接模式的核心思想是将抽象部分与实现部分分离,让它们都可以独立变化。这就好比建造一座桥,把两个独立的部分连接起来。在我们的电视机例子中,遥控器就是连接用户操作和电视机的"桥梁"。
在这里插入图片描述
如上图所示,桥接模式由四个核心角色组成:

  • 抽象类(Abstraction):定义抽象类的接口,它包含一个对实现类接口的引用。就像遥控器的抽象类,定义了基本的控制功能,并持有对具体电视机的引用。
  • 扩展抽象类(RefinedAbstraction):扩展抽象类,加入更多的功能特性。比如在基础遥控器的基础上,添加了更多高级功能的智能遥控器。
  • 实现类接口(Implementor):定义实现类的接口,供抽象类调用。就像所有品牌电视机都应该遵循的标准接口,定义了开关机、调节音量等基本功能。
  • 具体实现类(ConcreteImplementor):实现实现类接口的具体类。例如索尼电视机、三星电视机等具体品牌的电视机实现。

这种设计的妙处在于:如果我们要添加一种新的遥控器类型(比如带语音控制的遥控器),只需要扩展抽象类;如果要支持一个新品牌的电视机,只需要添加具体实现类。两个维度的变化互不影响,大大提高了系统的灵活性和可扩展性。

桥接模式案例实现

让我们通过遥控器控制电视机的完整示例,来深入理解桥接模式的实现。
在这里插入图片描述
首先,我们定义电视机的接口:

public interface TV {void on();void off();void tuneChannel(int channel);
}

然后实现具体品牌的电视机:

public class SonyTV implements TV {@Overridepublic void on() {System.out.println("索尼电视机启动");}@Overridepublic void off() {System.out.println("索尼电视机关闭");}@Overridepublic void tuneChannel(int channel) {System.out.println("索尼电视切换到" + channel + "频道");}
}public class SamsungTV implements TV {@Overridepublic void on() {System.out.println("三星电视机启动");}@Overridepublic void off() {System.out.println("三星电视机关闭");}@Overridepublic void tuneChannel(int channel) {System.out.println("三星电视切换到" + channel + "频道");}
}

接下来定义遥控器的抽象类:

public abstract class RemoteControl {protected TV tv;public RemoteControl(TV tv) {this.tv = tv;}public abstract void turnOn();public abstract void turnOff();
}

实现具体的遥控器类:

public class BasicRemote extends RemoteControl {public BasicRemote(TV tv) {super(tv);}@Overridepublic void turnOn() {System.out.println("基础遥控器:打开电视");tv.on();}@Overridepublic void turnOff() {System.out.println("基础遥控器:关闭电视");tv.off();}
}public class AdvancedRemote extends RemoteControl {public AdvancedRemote(TV tv) {super(tv);}@Overridepublic void turnOn() {System.out.println("高级遥控器:打开电视");tv.on();}@Overridepublic void turnOff() {System.out.println("高级遥控器:关闭电视");tv.off();}// 高级遥控器特有的功能public void setChannel(int channel) {System.out.println("高级遥控器:切换频道");tv.tuneChannel(channel);}
}

下面通过一个具体的客户端代码来演示如何使用这个遥控器-电视机系统:

public class Client {public static void main(String[] args) {// 创建不同品牌的电视TV sonyTV = new SonyTV();TV samsungTV = new SamsungTV();// 创建不同类型的遥控器RemoteControl basicRemote = new BasicRemote(sonyTV);AdvancedRemote advancedRemote = new AdvancedRemote(samsungTV);// 使用基础遥控器操作索尼电视System.out.println("使用基础遥控器操作索尼电视:");basicRemote.turnOn();basicRemote.turnOff();System.out.println("使用高级遥控器操作三星电视:");advancedRemote.turnOn();advancedRemote.setChannel(5);advancedRemote.turnOff();}
}

运行结果如下:
在这里插入图片描述
让我们通过下面的时序图来详细说明对象之间的交互过程:
在这里插入图片描述
从时序图中我们可以清晰地看到,当客户端调用遥控器的turnOn()方法时,遥控器会将请求转发给具体的电视机实现类。这种方式让遥控器和电视机之间形成了一个优雅的"桥接",它们可以独立变化而互不影响。

桥接模式的优缺点

优点

  • 很好地实现了抽象和实现的分离。就像我们的例子中,遥控器的功能和电视机的实现是完全分离的,这让我们能够独立地扩展任意一边而不会影响另一边。你可以添加一个带有语音控制功能的遥控器,而不需要修改任何电视机的代码。
  • 提高了系统的可扩展性。当系统需要添加新的实现时,只需要增加新的类,而不需要修改现有的代码。比如要支持新品牌的电视机,只需要实现TV接口即可,不需要修改任何遥控器的代码。
  • 对客户端来说很友好。客户端代码只需要知道抽象部分的接口,而不需要了解具体实现的细节。在我们的例子中,使用遥控器的人只需要知道按键的功能,而不需要关心不同品牌电视机的具体实现差异。

缺点

  • 增加了系统的复杂度。为了实现抽象和实现的分离,我们需要额外的接口和类。对于简单的系统来说,使用桥接模式可能会显得有点过度设计。
  • 要求正确识别系统中的两个独立变化的维度。如果对系统的理解不够深入,可能会导致对抽象和实现的划分不当,反而增加系统的维护难度。

桥接模式的适用场景

  • 当一个系统需要在抽象和实现之间具有灵活性时。例如,我们的遥控器和电视机的例子,两者之间的组合是动态的,任何遥控器都可以操控任何品牌的电视机。
  • 当系统中有两个或多个独立变化的维度时。比如除了遥控器和电视机的例子,类似的场景还有:形状和颜色(可以有不同形状和不同颜色的组合)、平台和视图(可以在不同平台上实现不同的视图)等。
  • 当需要跨平台或跨数据库时。桥接模式可以让具体的平台实现和业务逻辑分离,使得系统可以轻松地支持新的平台。
  • 当需要在运行时切换不同的实现时。由于抽象和实现是分离的,我们可以在运行时动态地改变实现,而不会影响到使用抽象接口的代码。

总结

桥接模式通过将抽象部分与实现部分分离,实现了两个维度上的独立扩展。就像遥控器和电视机的例子展示的那样,它让我们能够在不修改现有代码的情况下,轻松地添加新的遥控器类型或支持新品牌的电视机。
在实际应用中,使用桥接模式时需要注意以下几点:首先,要准确识别系统中独立变化的维度;其次,要合理划分抽象和实现的界限;最后,要避免过度设计,对于简单的系统,使用过于复杂的模式反而会增加开发和维护的成本。
桥接模式的精髓在于"分离抽象与实现",这种分离让系统更加灵活,更容易扩展和维护。当你遇到类似的场景时,不妨考虑使用桥接模式来优化你的设计。
在这里插入图片描述

关注犬余,共同进步

技术从此不孤单

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

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

相关文章

ASRPRO学习笔记一之语音模型位置和语音替换

一、语音替换的步骤 1、扬声器录音 打开GoldWave,点击工具栏中的蓝色控制属性按钮,点击设备,选择扬声器,点击ok。打开电脑上的网易云音乐,点击红色的录制按钮,开始录制音乐,在网易云音乐上点击播放音乐,录…

多因子认证 (Multi-factor authentication, MFA)

多因子认证 (MFA) 是一种思想,而UsernamePassword,OTP等是具体的认证手段。多因子认证就是将这些认证手段结合。 目录 什么是MFAMFA的作用MFA的实际应用 认证认证 (Authentication, AuthN) 因素常见的认证 (Authentication, AuthN) 类型密码认证无密码认…

内存压缩禁用设置

设置禁用内存压缩功能 1、“Win”“X”键→“A”键 2、如果输入“Get-MMAgent”并按“Enter”键,则可以从“MemoryCompression”中检查内存压缩功能的状态。 True启用,False禁用 3、要禁用内存压缩功能,请输入“Disable-MMAgent -mc”并…

素数回文数的个数

素数回文数的个数 C语言代码C 代码Java代码Python代码 💐The Begin💐点点关注,收藏不迷路💐 求11到n之间(包括n),既是素数又是回文数的整数有多少个。 输入 一个大于11小于1000的整数n。 输出…

QT网络(一):主机信息查询

网络简介 在QT中进行网络通信可以使用QT提供的Qt Network模块,该模块提供了用于编写TCP/IP网络应用程序的各种类,如用于TCP通信的QTcpSocket和 QTcpServer,用于 UDP 通信的 QUdpSocket,还有用于网络承载管理的类,以及…

Flutter Navigator2.0的原理和Web端实践

01 背景与动机 在Navigator 2.0推出之前,Flutter主要通过Navigator 1.0和其提供的 API(如push(), pop(), pushNamed()等)来管理页面路由。然而,Navigator 1.0存在一些局限性,如难以实现复杂的页面操作(如移…

如何在谷歌浏览器中开启安全浏览

在数字化时代,网络安全变得愈发重要。作为全球最受欢迎的网络浏览器之一,谷歌浏览器提供了多种功能来保护用户的在线安全。本文将详细介绍如何在谷歌浏览器中开启安全浏览,并额外提供一些有用的页面滚动设置、地址栏快捷搜索和跟踪防护的相关…

【物联网技术与应用】实验3:七彩LED灯闪烁

实验3 七彩LED灯闪烁 【实验介绍】 七彩LED灯上电后,7色动闪光LED模块可自动闪烁内置颜色。它可以用来制作相当吸引人的灯光效果。 【实验组件】 ● Arduino Uno主板* 1 ● USB数据线* 1 ● 7彩LED模块*1 ● 面包板*1 ● 9V方型电池*1 ● 跳线若干 【实验原…

直播美颜插件开发全流程:从美颜sdk的集成到实际部署

对于开发者来说,如何高效地开发和部署一个直播美颜插件,则需要从美颜SDK的集成到实际部署的全流程把控。本篇文章,我将详细解析这个流程中的关键技术和核心环节,助力开发者高效完成项目交付。 一、项目需求分析与技术选型 在开发…

【物联网技术与应用】实验4:继电器实验

实验4 继电器实验 【实验介绍】 继电器是一种用于响应施加的输入信号而在两个或多个点或设备之间提供连接的设备。换句话说,继电器提供了控制器和设备之间的隔离,因为设备可以在AC和DC上工作。但是,他们从微控制器接收信号,因此…

ESP32-S3外接SSD1306 OLED显示8*8字符和16*16汉字

一、接线图 二、实物 三、代码 #include <stdio.h> #include <string.h> #include "unity.h" #include "driver/i2c_master.h" #include "driver/gpio.h" #include "esp_lcd_panel_io.h" #include "esp_lcd_pane…

【Qt】QWidget中的常见属性及其功能(二)

目录 六、windowOpacity 例子&#xff1a; 七、cursor 例子&#xff1a; 八、font 九、toolTip 例子&#xff1a; 十、focusPolicy 例子&#xff1a; 十一、styleSheet 计算机中的颜色表示 例子&#xff1a; 六、windowOpacity opacity是不透明度的意思。 用于设…

Nginx Proxy Manager如何管理与配置反向代理服务并实现远程访问

文章目录 前言1. 一键安装2. 本地访问3. Linux 安装cpolar4. 配置公网访问地址5. 公网远程访问6. 固定公网地址 前言 Nginx Proxy Manager 是一个开源的反向代理工具&#xff0c;不需要了解太多 Nginx 或 Letsencrypt 的相关知识&#xff0c;即可快速将你的服务暴露到外部环境…

vscode中同时运行两个python文件(不用安装插件)

如何在vscode中同时运行两个python文件呢&#xff1f;今天在工作中遇到了这个问题。 查了网上的方法是安装coder runner插件&#xff0c;后来发现自身就有这个功能。所以记录一下,方便后续查找: 这是我的第一个文件&#xff0c;点击右上角的运行旁边的小箭头&#xff0c;有一…

Visio——导出的PDF文件缺乏嵌入字体的解决办法 / 设置导出的PDF文件添加嵌入字体的方法

导出PDF时&#xff0c;勾选 “符合PDF/A” 选项 这样就导出的PDF文件添加了嵌入字体了。

皮肤伤口分割数据集labelme格式248张5类别

数据集格式&#xff1a;labelme格式(不包含mask文件&#xff0c;仅仅包含jpg图片和对应的json文件) 图片数量(jpg文件个数)&#xff1a;284 标注数量(json文件个数)&#xff1a;284 标注类别数&#xff1a;5 标注类别名称:["bruises","burns","cu…

cpolar使用步骤

功能&#xff1a;内网穿透 下载地址&#xff1a;cpolar - secure introspectable tunnels to localhost 1 找到安装目录 2 进入命令行 目录处输入 cmd 3 验证 authtoken 不同用户 验证码不同。 注册后可以使用 cpolar.exe authtoken MzBlNzMwODktZjA3Yi00ZjJlLWJiMzQtNWU…

模具制造之三维扫描和逆向建模

模具是在工业生产中&#xff0c;用各种压力机和装在压力机上的专用工具&#xff0c;通过压力把金属或非金属材料制出所需形状的零件或制品&#xff0c;这种专用工具称为模具。模具的形状决定着这些产品的外形&#xff0c;模具的加工质量与精度也就决定着这些产品的质量。 汽车挡…

压力测试Jmeter简介

前提条件&#xff1a;要安装JDK 若不需要了解&#xff0c;请直接定位到左侧目录的安装环节。 1.引言 在现代软件开发中&#xff0c;性能和稳定性是衡量系统质量的重要指标。为了确保应用程序在高负载情况下仍能正常运行&#xff0c;压力测试变得尤为重要。Apache JMeter 是一…

[Pro Git#4] 标签 | 理解 | 创建 | push

目录 一、理解标签 二、创建标签 三、操作标签 一、理解标签 标签定义&#xff1a;在Git中&#xff0c;标签&#xff08;tag&#xff09;是对某次提交&#xff08;commit&#xff09;的一个标识&#xff0c;相当于起了一个别名。应用场景示例&#xff1a; 在项目发布某个版…