当前位置: 首页 > article >正文

「软件设计模式」桥接模式(Bridge Pattern)

深入解析桥接模式:解耦抽象与实现的艺术

一、模式思想:正交维度的优雅解耦

桥接模式(Bridge Pattern)通过分离抽象(Abstraction)与实现(Implementation),使二者可以独立扩展变化。这种结构型设计模式完美解决了多维交叉继承导致的类爆炸问题,如同在不同维度之间架设沟通的桥梁。

核心设计原则:

  1. 优先组合而非继承
  2. 抽象层与实现层独立演化
  3. 运行时绑定实现细节

二、场景案例:跨平台图形界面库

假设我们需要开发一个支持Windows/Linux/MacOS的图形界面库,包含按钮、输入框等控件。传统继承方式会导致:

AbstractControl
├── WindowsButton
├── LinuxButton
├── MacButton
├── WindowsInput
├── LinuxInput
└── MacInput

当新增控件类型或操作系统支持时,类数量将呈乘积增长。这正是桥接模式的用武之地。

三、模式结构解析

桥接模式结构图

关键角色:

  • 抽象化角色(Abstraction):定义高层控制逻辑
  • 扩展抽象化(Refined Abstraction):扩展的抽象接口
  • 实现化接口(Implementor):定义底层实现接口
  • 具体实现化(Concrete Implementor):具体的实现类

四、C++代码实现

#include <iostream>
#include <memory>// 实现化接口:操作系统图形API
class OSGraphicsAPI {
public:virtual ~OSGraphicsAPI() = default;virtual void drawButton(float x, float y, float w, float h) = 0;virtual void drawInputBox(float x, float y, float w, float h) = 0;
};// 具体实现化:Windows实现
class WindowsAPI : public OSGraphicsAPI {
public:void drawButton(float x, float y, float w, float h) override {std::cout << "Windows按钮绘制: (" << x << "," << y << ") " << w << "x" << h << std::endl;}void drawInputBox(float x, float y, float w, float h) override {std::cout << "Windows输入框绘制: [" << x << "," << y << "] " << w << "x" << h << std::endl;}
};// 具体实现化:Linux实现
class LinuxAPI : public OSGraphicsAPI {
public:void drawButton(float x, float y, float w, float h) override {std::cout << "Linux按钮绘制: (" << x << "," << y << ") " << w << "x" << h << std::endl;}void drawInputBox(float x, float y, float w, float h) override {std::cout << "Linux输入框绘制: [" << x << "," << y << "] " << w << "x" << h << std::endl;}
};// 抽象化接口:UI控件
class UIControl {
protected:std::unique_ptr<OSGraphicsAPI> impl_;public:explicit UIControl(std::unique_ptr<OSGraphicsAPI> api) : impl_(std::move(api)) {}virtual ~UIControl() = default;virtual void render() = 0;
};// 扩展抽象化:按钮控件
class Button : public UIControl {float x_, y_, w_, h_;public:Button(std::unique_ptr<OSGraphicsAPI> api, float x, float y, float w, float h): UIControl(std::move(api)), x_(x), y_(y), w_(w), h_(h) {}void render() override {std::cout << "渲染按钮 => ";impl_->drawButton(x_, y_, w_, h_);}
};// 扩展抽象化:输入框控件
class InputBox : public UIControl {float x_, y_, w_, h_;public:InputBox(std::unique_ptr<OSGraphicsAPI> api, float x, float y, float w, float h): UIControl(std::move(api)), x_(x), y_(y), w_(w), h_(h) {}void render() override {std::cout << "渲染输入框 => ";impl_->drawInputBox(x_, y_, w_, h_);}
};// 使用示例
int main() {// Windows平台控件auto winButton = std::make_unique<Button>(std::make_unique<WindowsAPI>(), 10, 20, 100, 30);winButton->render();// Linux平台输入框auto linuxInput = std::make_unique<InputBox>(std::make_unique<LinuxAPI>(), 50, 80, 200, 25);linuxInput->render();return 0;
}

运行模式:

五、应用场景与优势

适用场景

  • 多维度独立扩展的系统(平台x功能,设备x驱动)
  • 需要运行时切换实现方案
  • 避免多层继承结构

独特优势

  1. 正交扩展性:新增维度只需添加对应层级的类
  2. 单一职责原则:抽象关注逻辑,实现专注细节
  3. 开闭原则:各层级独立扩展,无需修改已有代码

六、模式变体与演进

  • 嵌套桥接:多层桥接处理更多维度
  • 结合工厂方法:动态创建具体实现
  • 策略模式融合:运行时切换不同实现策略

七、性能考量与实践建议

虽然桥接模式通过间接调用带来一定性能开销,但现代计算机的优化能力使其几乎可以忽略。建议:

  1. 使用智能指针管理实现对象生命周期
  2. 优先采用接口组合而非多层继承
  3. 合理控制抽象层级,避免过度设计

八、总结

桥接模式为复杂系统提供了优雅的维度解耦方案,其核心价值在于:

  • 分离变与不变的部分
  • 建立抽象与实现的动态绑定
  • 提升系统的可维护性和扩展性

当系统出现正交维度的扩展需求时,桥接模式如同架设在抽象与实现之间的智能立交桥,让不同维度的变化能够各行其道,这正是优秀软件架构设计的精髓所在。

相关文章:

「软件设计模式」桥接模式(Bridge Pattern)

深入解析桥接模式&#xff1a;解耦抽象与实现的艺术 一、模式思想&#xff1a;正交维度的优雅解耦 桥接模式&#xff08;Bridge Pattern&#xff09;通过分离抽象&#xff08;Abstraction&#xff09;与实现&#xff08;Implementation&#xff09;&#xff0c;使二者可以独立…...

Python 用户输入和While循环(使用while 循环来处理列表和字典)

大多数程序都旨在解决最终用户的问题&#xff0c;为此通常需要从用户那里获取一些信息。例如&#xff0c;假设有人要判断自己是否到了投票的年龄&#xff0c;要编写回答这个问题的程序&#xff0c;就 需要知道用户的年龄&#xff0c;这样才能给出答案。因此&#xff0c;这种程序…...

docker 基础命令使用(ubuntu)

docker 状态查询 docker ps docker ps -adocker --version docker info docker --help docker run --help docker ps --help ...docker 操作镜像命令 docker imagesdocker rmi 镜像id/镜像名docker 操作容器命令 docker ps docker ps -adocker run 命令 # 端口映射 -p 参数…...

Jenkins 安装插件 二

Jenkins 安装插件 二 一. 打开 Dashboard 打开 Jenkins 界面&#xff0c;不管在任何界面&#xff0c;只需要点击左上角 Dashboard 按钮即可 二. 打开 Manage Jenkins 找到 Manage Jenkins -> System Configuration -> Plugins 点击 Plugins 打开界面如下 Updates&a…...

深入解析与解决 Oracle 报错:ORA-29275 部分多字节字符20250213

&#x1f6e0;️ 深入解析与解决 Oracle 报错&#xff1a;ORA-29275 部分多字节字符 引言 &#x1f31f; 在与 Oracle 数据库打交道的日常工作中&#xff0c;你是否遇到过 ORA-29275: partial multibyte character 这个令人头疼的错误&#xff1f;这个错误通常与字符编码、数…...

CI/CD(二)docker-compose安装Jenkins

1、docker-compose.yml version: 3.8services:jenkins:image: jenkins/jenkins:lts # 使用官方的 Jenkins LTS 镜像container_name: jenkinsuser: root # 如果需要以 root 用户运行ports:- "8080:8080" # Jenkins Web 界面端口- "50000:50000" # 用于 Jen…...

Windows环境安装部署minimind步骤

Windows环境安装部署minimind步骤 必要的软件环境 git git&#xff0c;可下载安装版&#xff0c;本机中下载绿色版&#xff0c;解压到本地目录下&#xff08;如&#xff1a;c:\soft\git.win64&#xff09;&#xff0c;可将此路径添加到PATH环境变量中&#xff0c;供其他程序…...

使用Node.js进行串口通信

目录 一、 安装 serialport 库二.、实现方法1.打开串口并配置参数2. 向串口传递信息3. 接收串口信息4. 处理错误5. 关闭串口6. 使用解析器7. 获取串口列表 三、 完整示例代码 一、 安装 serialport 库 首先&#xff0c;需要安装 serialport 库。可以通过 npm 安装&#xff1a;…...

C# ASP.NET的应用场景

.NET学习资料 .NET学习资料 .NET学习资料 C# ASP.NET作为一种强大的 Web 开发框架&#xff0c;在众多领域都有着广泛的应用&#xff0c;为各类 Web 应用的开发提供了高效、可靠的解决方案。以下是其主要的应用场景&#xff1a; 企业级 Web 应用 在企业级应用开发中&#xf…...

常用的网络安全设备

大家读完觉得有帮助&#xff0c;记得关注和点赞&#xff01;&#xff01;&#xff01; 一、 WAF 应用防火墙 范围&#xff1a;应用层防护软件 作用&#xff1a; 通过特征提取和分块检索技术进行模式匹配来达到过滤&#xff0c;分析&#xff0c;校验网络请求包的目的&#x…...

Qt信号槽调用出错:Qt: Dead lock detected while activating a BlockingQueuedConnection

目录 1.现象和原因分析 2. 总结 1.现象和原因分析 就在最近的开发过程中&#xff0c;程序一运行在控制台就打印&#xff1a; Qt: Dead lock detected while activating a BlockingQueuedConnection&#xff1a; 咋一看&#xff0c;怎么出现死锁了呢&#xff1f;仔细看下…...

应对DeepSeek总是服务器繁忙的解决方法

最近由于访问量过大&#xff0c;DeepSeek服务器官网经常弹出&#xff1a;“服务器繁忙&#xff0c;请稍后再试”的提示&#xff0c;直接卡成PPT怎么办&#xff1f;服务器繁忙直接看到视觉疲劳&#xff1a; 解决DeepSeek卡顿问题 DeepSeek使用卡顿问题&#xff0c;是因为访问量…...

【ISO 14229-1:2023 UDS诊断全量测试用例清单系列:第八节】

ISO 14229-1:2023 UDS诊断服务测试用例全解析&#xff08;RequestUpload0x35服务&#xff09; 作者&#xff1a;车端域控测试工程师 更新日期&#xff1a;2025-02-13 关键词&#xff1a;UDS协议、0x35服务、数据上传、内存读取、ECU测试 一、服务功能概述 0x35服务&#xff08…...

仿叮咚买菜鸿蒙原生APP

# DingdongShopping 这是一个原生鸿蒙版的仿叮咚买菜APP项目 鸿蒙Next发布至今已经有一年多的时间了&#xff0c;但有时候我们想要实现一些复杂的功能或者效果&#xff0c;在开发文档上查阅一些资料还是比较费时的&#xff0c;有可能还找不到我们想要的内容。而社会层面上分享…...

【kafka系列】Kafka事务的实现原理

目录 1. 事务核心组件 1.1 幂等性生产者&#xff08;Idempotent Producer&#xff09; 1.2 事务协调器&#xff08;TransactionCoordinator&#xff09; 1.3 事务日志&#xff08;Transaction Log&#xff09; 2. 事务执行流程 2.1 事务初始化 2.2 发送消息 2.3 事务提…...

HarmonyOS NEXT网络状态监听HTTP和RCP请求网络

当我们在HarmonyOS NEXT中开发的应用&#xff0c;基本上都会使用网络请求&#xff0c;从服务端获取数据在客户端显示或者供用户交互&#xff0c;有时候网络发生变化时&#xff0c;我们需要做一些相应的操作&#xff0c;接下来我们一起来了解下在HarmonyOS NEXT下如何监听网络状…...

2025.2.16

Web [GDOUCTF 2023]泄露的伪装&#xff1a; 点进去看就是装神弄鬼&#xff0c;那就直接扫描 果然有东西 第一个是php代码 第二个是个文件 访问发现是一样的 分析一下&#xff1a;使用 file_get_contents($cxk) 函数读取 $cxk 变量中指定的 URL 或文件的内容。 如果读取的内…...

使用Java爬虫获取京东JD.item_sku API接口数据

在电商领域&#xff0c;商品的SKU&#xff08;Stock Keeping Unit&#xff09;信息是运营和管理的关键数据。SKU信息包括商品的规格、价格、库存等&#xff0c;对于商家的库存管理、定价策略和市场分析至关重要。京东作为国内领先的电商平台&#xff0c;提供了丰富的API接口&am…...

【ISO 14229-1:2023 UDS诊断(会话控制0x10服务)测试用例CAPL代码全解析③】

ISO 14229-1:2023 UDS诊断【会话控制0x10服务】_TestCase03 作者&#xff1a;车端域控测试工程师 更新日期&#xff1a;2025年02月15日 关键词&#xff1a;UDS诊断、0x10服务、诊断会话控制、ECU测试、ISO 14229-1:2023 TC10-003测试用例 用例ID测试场景验证要点参考条款预期…...

MySQL安装MySQL服务时提示Install-Remove of the Service Denied

文章目录 问题描述排查1.字面意思2.搜索引擎3.官方文档4.源码 处理方法相关扩展 问题描述 MySQL安装MySQL服务时提示Install-Remove of the Service Denied! 详细报错如下&#xff1a; C:\Users\荷塘月色>net start mysql 服务名无效。请键入 NET HELPMSG 2185 以获得更多…...

python学opencv|读取图像(六十五)使用cv2.boundingRect()函数实现图像轮廓矩形标注

【1】引言 前序学习进程中&#xff0c;已经使用cv2.findContours()函数cv2.drawContours()函数实现图像轮廓识别和标注&#xff0c;这种标注沿着图像的轮廓进行&#xff0c;比较细致。相关文章链接为&#xff1a; python学opencv|读取图像&#xff08;六十四&#xff09;使用…...

haproxy实现MySQL服务器负载均衡

1.环境准备 准备好下面四台台服务器&#xff1a; 主机名IP角色open-Euler1192.168.121.150mysql-server1openEuler-2192.168.121.151mysql-server2openEuler-3192.168.121.152clientRocky8-1192.168.121.160haproxy 2.mysql服务器配置 1.下载mariadb #下载mariadb [rootop…...

open3d绘制平面

在Open3D中绘制平面通常涉及到创建一个平面模型并将其可视化。Open3D是一个开源库,主要用于3D数据的处理和可视化,但它主要用于3D数据的处理,并不直接支持绘制2D平面。如果你想在Open3D中“绘制”一个平面,你可以通过以下几种方法来实现类似的效果: 方法1:使用o3d.geome…...

【ISO 14229-1:2023 UDS诊断全量测试用例清单系列:第十四节】

ISO 14229-1:2023 UDS诊断服务测试用例全解析&#xff08;CommunicationControl_0x28服务&#xff09; 作者&#xff1a;车端域控测试工程师 更新日期&#xff1a;2025年02月14日 关键词&#xff1a;UDS协议、0x28服务、通信控制、ISO 14229-1:2023、ECU测试 一、服务功能概述…...

C语言简单练习题

文章目录 练习题一、计算n的阶乘bool类型 二、计算1!2!3!...10!三、计算数组arr中的元素个数二分法查找 四、动态打印字符Sleep()ms延时函数system("cls")清屏函数 五、模拟用户登录strcmp()函数 六、猜数字小游戏产生一个随机数randsrandRAND_MAX时间戳time() 示例 …...

每日一题——把数字翻译成字符串

把数字翻译成字符串 题目描述示例示例1示例2 题解动态规划代码实现复杂度分析 总结 题目描述 有一种将字母编码成数字的方式&#xff1a;‘a’->1, ‘b’->2, … , ‘z’->26。 现在给一串数字&#xff0c;返回有多少种可能的译码结果。 数据范围&#xff1a;字符串…...

C语言之easyX

目录 概要 easyX整体架构 图形绘制 画布宽高 圆形 图片的贴图 加载图像 游戏框架 概要 easyX是一个轻量级的图形库&#xff0c;用于在Windows平台上进行简单的2D图形绘制。它提供了一组简单易用的函数&#xff0c;可以方便地绘制基本的图形元素&#xff0c;如线条、矩形、圆形…...

安卓基础(持续更新的笔记)

为什么要这样&#xff1a; // 创建请求体RequestBody body RequestBody.create(MediaType.parse("application/json; charsetutf-8"),jsonObject.toString()); jsonObject 就包含了一个 JSON 数据&#xff0c;它其实就是&#xff1a; {"name": "张…...

10. Hbase Compaction命令

一. 什么是Compaction 在 HBase 中&#xff0c;频繁进行数据插入、更新和删除操作会生成许多小的 HFile&#xff0c;当 HFile 数量增多时&#xff0c;会影响HBase的读写性能。此外&#xff0c;垃圾数据的存在也会增加存储需求。因此&#xff0c;定期进行 Compact操作&#xff…...

在 UniApp 项目中设置多语言

在 UniApp 项目中设置多语言支持可以通过以下步骤实现&#xff1a; 1. 安装依赖 首先&#xff0c;你需要安装 vue-i18n 插件来处理多语言支持。 npm install vue-i18n --save2. 创建语言文件 在项目中创建一个 lang 文件夹&#xff0c;用于存放不同语言的翻译文件。例如&am…...