结构型设计模式之桥接模式【设计模式系列】

news2024/11/24 20:38:04

系列文章目录

C++技能系列
Linux通信架构系列
C++高性能优化编程系列
深入理解软件架构设计系列
高级C++并发线程编程
设计模式系列

期待你的关注哦!!!
在这里插入图片描述

现在的一切都是为将来的梦想编织翅膀,让梦想在现实中展翅高飞。
Now everything is for the future of dream weaving wings, let the dream fly in reality.

结构型设计模式之桥接模式

  • 系列文章目录
  • 一、桥接模式介绍
  • 二、桥接模式优缺点
    • 2.1 优点
    • 2.2 缺点
  • 三、桥接模式使用场景
  • 四、桥接模式实现
  • 五、桥接模式应用实例

一、桥接模式介绍

⚠️ 意图:
“将抽象化(Abstraction)与实现化(Implementation)脱耦,使得二者可以独立地变化”。这句话有三个关键词,也就是抽象化实现化脱耦

⚠️ 主要解决:
在有多种可能会变化的情况下,用继承会造成类爆炸问题,扩展起来不灵活。

⚠️ 何时使用:
实现系统可能有多个角度分类,每一种角度都可能变化。

⚠️ 如何解决:
把这种多角度分类分离出来,让它们独立变化,减少它们之间耦合。将继承关系简化为组合关系。

桥接模式中的实现不是指抽象基类的具体子类对抽象基类中虚函数(接口)的实现,是指怎么去实现用户的需求。即在Implement具体类中实现Abstraction的接口功能,并且是通过组合(委托)的方式实现的,因此桥接模式中实现不是指的继承基类、实现基类接口,而是指的是通过对象组合实现用户的需求。
在这里插入图片描述

图1_1 桥接模式类图

桥接模式将继承关系转换为组合关系,从而降低了系统间的耦合,减少了代码编写量。使用组合(委托)的方式将抽象和实现彻底地解耦,好处是抽象和实现可以分别独立地变化,系统的耦合性也得到了很好的降低。

将抽象部分与它的实现部分分离,使得它们可以独立地变化。抽象Abstraction与实现Implement分离,抽象部分Abstraction可以变化,如new RefinedAbstractionA(imp)、new RefinedAbstractionB(imp);实现部分Implement也可以独立变化,如new ConcreteImplementA()、new ConcreteImplementB()。

二、桥接模式优缺点

2.1 优点

  • 分离接口及其实现部分 。一个实现未必不变地绑定在一个接口上。抽象类的实现可以在运行时刻进行配置,一个对象甚至可以在运行时刻改变它的实现。将Abstraction与Implement分离有助于降低对实现部分编译时刻的依赖性,当改变一个实现类时,并不需要重新编译Abstraction类和它的客户程序。为了保证一个类库的不同版本之间的二进制兼容性,一定要有这个性质。另外,接口与实现分离有助于分层,从而产生更好的结构化系统,系统的高层部分仅需知道Abstraction和Implement即可。

  • 提高可扩充性 。可以独立地对Abstraction和Implement层次结构进行扩充。

  • 实现细节对客户透明 。可以对客户隐藏实现细节,例如共享Implement对象以及相应的引用计数机制。

  • 将可以共享的变化部分,抽离出来,减少了代码的重复信息。

  • 对象的具体实现可以更加灵活,可以满足多个因素变化的要求。

2.2 缺点

  • 桥接模式的引入会增加系统的理解与设计难度,由于聚合关联关系建立在抽象层,要求开发者针对抽象进行设计与编程。

  • 桥接模式要求正确识别出系统中两个独立变化的维度,因此其使用范围具有一定的局限性。

  • 客户必须知道选择哪一种类型的实现

三、桥接模式使用场景

  • 当一个对象有多个变化因素的时候,考虑依赖于抽象的实现,而不是具体的实现。如手机品牌有2种变化因素,一个是品牌,一个是功能。

  • 当多个变化因素在多个对象间共享时,考虑将变化的部分抽象出来再聚合或组合进来。

  • 当考虑一个对象的多个变化因素可以动态变化的时候,考虑使用桥接模式,如手机的手机品牌是变化的,手机的功能也是变化的,所以将每个可变化的因数分离出来,独立的变化。

  • 抽象工厂模式可以用来创建和配置一个特定的桥接模式。

  • 适配器模式用来帮助无关的类协同工作,通常在系统设计完成后才会被使用。然而,桥接模式则是在系统开始时就被使用,使得抽象接口和实现部分可以独立进行改变。

  • 桥接模式和装饰模式在一定程度上都是为了减少子类的数目,避免出现复杂的继承关系,但解决的方法却各有不同。装饰模式把子类中比基类中多出来的部分放到单独的类里面,以适应新功能增加的需要,当把描述新功能的类封装到基类的对象里面时,就得到所需要的子类对象,描述新功能的类通过组合可以实现很多的功能组合。 桥接模式则把原来的基类的实现化细节抽象出来,在构造到一个实现化的结构中,然后再把原来的基类改造成一个抽象化的等级结构,就可以实现系统在多个维度上的独立变化 。

四、桥接模式实现

Abstraction抽象基类:

#ifndef BRIDGE_ABSTRACTION_H
#define BRIDGE_ABSTRACTION_H

#include "Implement.h"

//抽象基类
class Abstraction {
public:
    //需要实现的接口
    virtual void Request() = 0;

protected:
    explicit Abstraction(Implement* imp):implement_(imp) {}

protected:
    Implement *implement_;
};

#endif //BRIDGE_ABSTRACTION_H

RefinedbstractionA具体类:

#ifndef BRIDGE_REFINEDBSTRACTIONA_H
#define BRIDGE_REFINEDBSTRACTIONA_H

#include <iostream>
#include "Abstraction.h"
#include "Implement.h"

using namespace std;

class RefinedbstractionA : public Abstraction{
public:
    explicit RefinedbstractionA(Implement* imp) : Abstraction(imp){}
    ~RefinedbstractionA() = default;
    void Request() override  {
        cout <<  "RefinedbstractionA::Request" << endl;
        //调用实现部分
        implement_->OperationImpl();
    }
};


#endif //BRIDGE_REFINEDBSTRACTIONA_H

RefinedbstractionB具体类:

#ifndef BRIDGE_REFINEDBSTRACTIONB_H
#define BRIDGE_REFINEDBSTRACTIONB_H

#include <iostream>
#include "Abstraction.h"
#include "Implement.h"

using namespace std;

class RefinedbstractionB : public Abstraction{
public:
    explicit RefinedbstractionB(Implement* imp) : Abstraction(imp){}
    ~RefinedbstractionB() = default;
    void Request() override  {
        cout <<  "RefinedbstractionB::Request" << endl;
        //调用实现部分
        implement_->OperationImpl();
    }
};

#endif //BRIDGE_REFINEDBSTRACTIONB_H

Implement抽象实现类:

#ifndef BRIDGE_IMPLEMENT_H
#define BRIDGE_IMPLEMENT_H

//抽象实现类
class Implement {
protected:
    Implement() = default;
    ~Implement() = default;

public:
    virtual void OperationImpl() = 0;
};


#endif //BRIDGE_IMPLEMENT_H

ConcreteImplementA具体实现类:

#ifndef BRIDGE_CONCRETEIMPLEMENTA_H
#define BRIDGE_CONCRETEIMPLEMENTA_H

#include <iostream>
#include "Implement.h"

using namespace std;
//具体实现类
class ConcreteImplementA : public Implement{
public:
    ConcreteImplementA() = default;
    ~ConcreteImplementA() = default;
    //具体实现的功能函数
    void OperationImpl() override{
        cout <<  "ConcreteImplementA::OperationImpl" << endl;
    }
};


#endif //BRIDGE_CONCRETEIMPLEMENTA_H

ConcreteImplementB具体实现类:

#ifndef BRIDGE_CONCRETEIMPLEMENTB_H
#define BRIDGE_CONCRETEIMPLEMENTB_H


#include <iostream>
#include "Implement.h"

using namespace std;
//具体实现类
class ConcreteImplementB : public Implement{
public:
    ConcreteImplementB() = default;
    ~ConcreteImplementB() = default;
    //具体实现的功能函数
    void OperationImpl() override{
        cout <<  "ConcreteImplementB::OperationImpl" << endl;
    }
};


#endif //BRIDGE_CONCRETEIMPLEMENTB_H

BridgeMain使用类:

#include <iostream>
#include "Abstraction.h"
#include "Implement.h"
#include "ConcreteImplementA.h"
#include "ConcreteImplementB.h"
#include "RefinedbstractionA.h"
#include "RefinedbstractionB.h"


int main() {

    Implement* imp = new ConcreteImplementA();
    Abstraction* abs = new RefinedbstractionA(imp);
    abs->Request();

    Implement* imp1 = new ConcreteImplementB();
    Abstraction* abs1 = new RefinedbstractionB(imp1);
    abs1->Request();


    return 0;
}

将抽象部分与实现部分分离:实现系统可能有多角度(维度)分类,每一种分类都可能变化,把多种角度分离出来让它们独立变化,减少它们之间的耦合。

在发现需要多角度去分类实现对象,而只用继承会造成大量的类增加,不能满足开放-封闭原则时,要考虑用桥接模式。

组合/聚合复用原则:尽量使用组合/聚合,不要使用类继承。

优先使用对象的组合/聚合将有助于保持每个类被封装,并被集中在单个任务上。类和类继承层次会保持较小规模,并且不太可能增长为不可控制的庞然大物。

五、桥接模式应用实例

电脑品牌和操作系统是两个概念,不同的品牌的电脑可以安装相同或不同的操作系统,两者都具有很大的变动性。如果单独以电脑品牌或操作系统为基类来进行继承扩展的话,会使类的数目剧增并且耦合性很高,如果更改电脑品牌或增加操作系统类型都会增加很多的变动。

将两者抽象出来两个基类分别是Computer和OS,在Computer类中聚合一个OS对象的基类将解决电脑品牌和操作系统扩展混乱的问题,两者的扩展就相对灵活,剪短了两者的必要联系。

Computer接口:

#ifndef COMPUTER_H
#define COMPUTER_H
 
class OS;
class Computer
{
public:
    virtual void installOS(OS* os) = 0;
};
 
#endif // COMPUTER_H

AppleComputer具体实现:

#ifndef APPLECOMPUTER_H
#define APPLECOMPUTER_H
#include "Computer.h"
#include "OS.h"
 
class AppleComputer : public Computer
{
public:
    virtual void installOS(OS* os)
    {
        cout << "AppleComputer ";
        os->installOS_Imp();
    }
};
 
#endif // APPLECOMPUTER_H

ThinkPadComputer具体实现:

#ifndef THINKPADCOMPUTER_H
#define THINKPADCOMPUTER_H
#include "Computer.h"
#include "OS.h"
 
class ThinkPadComputer : public Computer
{
public:
    virtual void installOS(OS* os)
    {
        cout << "ThinkPadComputer ";
        os->installOS_Imp();
    }
};
 
#endif // THINKPADCOMPUTER_H

OS接口:

#ifndef OS_H
#define OS_H
#include <iostream>
using namespace std;
 
class OS
{
public:
    virtual void installOS_Imp() = 0;
};
 
#endif // OS_H

LinuxOS具体实现:

#ifndef LINUXOS_H
#define LINUXOS_H
#include "OS.h"
 
class LinuxOS : public OS
{
public:
    virtual void installOS_Imp()
    {
        cout << "has installed Linux OS" << endl;
    }
};
 
#endif // LINUXOS_H

WindowsOS具体实现:

#ifndef WINDOWSOS_H
#define WINDOWSOS_H
#include "OS.h"
 
class WindowsOS : public OS
{
public:
    virtual void installOS_Imp()
    {
        cout << "has installed Windows OS" << endl;
    }
};
 
#endif // WINDOWSOS_H

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

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

相关文章

Vue3状态管理库Pinia——核心概念(Store、State、Getter、Action)

个人简介 &#x1f440;个人主页&#xff1a; 前端杂货铺 &#x1f64b;‍♂️学习方向&#xff1a; 主攻前端方向&#xff0c;正逐渐往全干发展 &#x1f4c3;个人状态&#xff1a; 研发工程师&#xff0c;现效力于中国工业软件事业 &#x1f680;人生格言&#xff1a; 积跬步…

行为型模式 - 迭代器模式

概述 定义&#xff1a; 提供一个对象来顺序访问聚合对象中的一系列数据&#xff0c;而不暴露聚合对象的内部表示。 结构 迭代器模式主要包含以下角色&#xff1a; 抽象聚合&#xff08;Aggregate&#xff09;角色&#xff1a;定义存储、添加、删除聚合元素以及创建迭代器对象…

Mind+积木编程控制小水泵给宠物喂水

前期用scratch&#xff0c;带着小朋友做了大鱼吃小鱼、桌面弹球、小学生计算器3个作品&#xff0c;小朋友收获不小。关键是小家伙感兴趣&#xff0c;做出来后给家人炫耀了一圈后&#xff0c;兴趣大增&#xff0c;嚷嚷着要做更好玩的。 最近&#xff0c;娃妈从抖音上买了个小猫喝…

JMeter 配置环境变量步骤

通过给 JMeter 配置环境变量&#xff0c;可以快捷的打开 JMeter&#xff1a; 打开终端。执行 jmeter。 配置环境变量的方法如下。 Mac 和 Linux 系统 1、在 ~/.bashrc 中加如下内容&#xff1a; export JMETER_HOMEJMeter所在目录 export PATH$JAVA_HOME/bin:$PATH:.:$JME…

pytorch安装GPU版本 (Cuda12.1)教程: Windows、Mac和Linux系统下GPU版PyTorch(CUDA 12.1)快速安装

&#x1f337;&#x1f341; 博主 libin9iOak带您 Go to New World.✨&#x1f341; &#x1f984; 个人主页——libin9iOak的博客&#x1f390; &#x1f433; 《面试题大全》 文章图文并茂&#x1f995;生动形象&#x1f996;简单易学&#xff01;欢迎大家来踩踩~&#x1f33…

【单调栈 +前缀和】AcWing 4738. 快乐子数组

原题链接 原题链接 相关算法概念介绍 前缀和&#xff08;Prefix Sum&#xff09; 前缀和是指将数组中从开头位置到当前位置的所有元素累加得到的新数组。通常&#xff0c;我们使用一个额外的数组来保存这些累加和&#xff0c;这个数组被称为前缀和数组。对于原始数组A&…

Appium+python自动化(十七)- - Monkey

1、Monkey简介 在Android的官方自动化测试领域有一只非常著名的“猴子”叫Monkey&#xff0c;这只“猴子”一旦启动&#xff0c;就会让被测的Android应用程序像猴子一样活蹦乱跳&#xff0c;到处乱跑。人们常用这只“猴子”来对被测程序进行压力测试&#xff0c;检查和评估被测…

快速排序QuickSort

目录 1.Hoare法 2.挖坑法 3.前后指针法 4.快排分治 5.关于快排 6.关于快排的优化 7.总体实现 总结&#xff1a; 快速排序是Hoare于1962年提出的一种二叉树结构的交换排序方法 其基本思想为&#xff1a;任取待排序元素序列中的某元素作为基准值&#xff0c;按照该排序码…

《5.linux驱动开发-第2部分-5.2.字符设备驱动基础》5.2.5.用开发板来调试模块

1. 首先 开发板 可以运行 Uboot 2. Ubuntu 安装好了 t f t p(启动内核zImage) 和 NFS &#xff08;挂载 根文件系统&#xff09; 3. 提前 制作好了 根文件系统&#xff08;2022年做的&#xff0c;早就忘记 怎么做了&#xff09; 4.内核 需要设置 nfs 作为根文件系统 启动…

聊聊spring-cloud的负载均衡

聊聊spring-cloud的负载均衡 1. 选择合适的负载均衡算法2. 合理设置超时时间3. 缓存服务实例列表4. 使用断路器5. 使用缓存Spring Cloud负载均衡组件对比RibbonLoadBalancerWebClient对比 总结 在微服务架构中&#xff0c;负载均衡是非常重要的一个环节&#xff0c;可以有效地提…

ES6基础知识三:对象新增了哪些扩展?

一、属性的简写 ES6中&#xff0c;当对象键名与对应值名相等的时候&#xff0c;可以进行简写 const baz {foo:foo}// 等同于 const baz {foo}方法也能够进行简写 const o {method() {return "Hello!";} };// 等同于const o {method: function() {return "…

C# List 详解四

目录 18.FindLast(Predicate) 19.FindLastIndex(Int32, Int32, Predicate) 20.FindLastIndex(Int32, Predicate) 21.FindLastIndex(Predicate) 22.ForEach(Action) 23.GetEnumerator() 24.GetHashCode() 25.GetRange(Int32, Int32) C#…

协作实现时序数据高效流转链路 | 7.20 IoTDB X RocketMQ 技术沙龙线上直播回顾

7 月 20 日&#xff0c;IoTDB X RocketMQ 技术沙龙线上直播圆满结束。工业物联网时序数据库研发商天谋科技、云原生事件流平台 Apache RocketMQ 社区的四位技术专家&#xff0c;针对实时数据接入、多样数据处理与系统的高扩展、高可靠特性的数据流转处理平台实现难点&#xff0…

计算机服务器被devos勒索病毒攻击怎么解决,数据库解密恢复方式

科学技术的发展为企业的生产运行提供了极大的便利性&#xff0c;但随之而来的网络安全也应该引起人们的重视。近期&#xff0c;我们收到很多企业的求助&#xff0c;企业的计算机服务器内的数据库被devos后缀勒索病毒攻击&#xff0c;导致企业许多工作无法正常运行。Devos后缀勒…

89、简述RabbitMQ的架构设计

简述RabbitMQ的架构设计 BrokerQueueExchangeRoutingKeyBinding信道架构设计图 Broker RabbitMQ的服务节点 Queue 队列&#xff0c;是RabbitMQ的内部对象&#xff0c;用于存储消息。RabbitMQ中消息只能存储在队列中。生产者投递消息到队列&#xff0c;消费者从队列中获取消息…

Sql Developer日期显示格式问题

sqldeveloper模式日期显示不是很美观 并且使用日期条件查询需要将月份转为中文&#xff0c;系统兼容性差 容易以前如下报错 ORA-01861: 文字与格式字符串不匹配 01861. 00000 - "literal does not match format string"-- sqldeveloper 中执行日期条件 &#xff08;…

2023杭电多校第二场1007-foreverlasting and fried-chicken

链接&#xff1a;Problem - 7293 (hdu.edu.cn) 思路&#xff1a; 枚举度大于4 和 6 且 共同连接 4个以上点 的两个点, 其度分别记为a 和 b 若a为上面的点, 那么答案为C(a-4, 2) * C(b, 4), 反之同理 如果直接搜点会tle, 此时用bitset优化, 状态压缩, 时间复杂度为O(n^3 /32) …

只需3步,使用Stable Diffusion无限生产AI数字人视频

效果演示 先看效果&#xff0c;感兴趣的可以继续读下去。 没有找到可以上传视频的地方&#xff0c;大家打开这个网盘链接观看&#xff1a;https://www.aliyundrive.com/s/CRBm5NL3xAE 基本方法 搞一张照片&#xff0c;搞一段语音&#xff0c;合成照片和语音&#xff0c;同…

APP抓包-资产获取+Frida反代理绕过和证书校验绕过

app抓包获取资产 1.打开模拟器&#xff0c;和电脑连接同一个wifi&#xff0c;让模拟器和电脑处于同一局域网&#xff0c;在模拟器配置代理。 burp开启监听 模拟器开启app&#xff0c;burp成功获取资产信息 有时候明明配置没问题&#xff0c;为什么抓不到app数据包呢&#xff1f…

Folx Pro 5 最好用的Mac磁力链接BT种子下载工具

除了迅雷&#xff0c;还有哪个支持磁力链接下载&#xff1f;Mac电脑如何下载磁力链接&#xff1f;经常有小伙伴问老宅。今天&#xff0c;老宅给大家推荐Folx Pro For Mac&#xff0c;Mac系统超好用的磁力下载工具。 Folx是一款功能强大且易于使用的Mac下载管理器&#xff0c;并…