C++设计模式:桥模式(五)

news2025/1/14 0:42:41
1、定义与动机
  • 桥模式定义:将抽象部分(业务功能)与实现部分(平台实现)分离,使他们可以独立地变化
  • 引入动机
    • 由于某些类型的固有的实现逻辑,使得它们具有两个变化的维度,乃至多个维度的变化
    • 如何应对这种“多维度的变化”?如何利用面向对象技术来使用类型可以轻松地沿着两个乃至多个方向变化,二部引入额外的复杂度?
2、案例分析

假设现在存在这样一个需求:

  • 有一个消息发送器Messager,里面有七个方法
    • 大致需要完成登录、发送文本消息、发送图片消息、播放声音、建立连接等
    • 同时由于存在不同的终端设备:例如手机、电脑、pad等,他们的这些实现方式肯定会存在一定的不同
    • 并且对于同一种设备存在不同的功能组合问题:例如发送图片、消息之前播放声音等。
2.1、首先设计一个抽象接口

根据上面所提出的需求可以写出如下的接口代码,函数都是纯虚函数和虚函数组成

class Messager{
public:
    virtual void Login() = 0;
    virtual void SendMessage() = 0;
    virtual void SendPicture() = 0;

    virtual void PlaySound() = 0;
    virtual void DrawShape() = 0;
    virtual void WriteText() = 0;
    virtual void Connect() = 0;

    virtual ~Messager(){

    }
};
2.2、针对不同的终端平台编码设计
  • 例如针对PC、手机…平台设计它们的是如何发送文本、画图、播放声音、建立网络通信连接的代码
  • 但是这些Base类依然是一个抽象基类,因为存在一些纯虚函数没有实现,因此其本身也是一个抽象类
// PC平台实现(抽象类),没有完全实现完毕所有的纯虚函数
class PCMessagerBase: public Messager{
public:
    virtual void PlaySound(){
        // *****
    }
    virtual void DrawShape(){
        // *****
    }
    virtual void WriteText(){
        // *****
    }
    virtual void Connect(){
        // *****
    }
}

// Mobile平台实现(抽象类),没有完全实现完毕所有的纯虚函数
class MobileMessagerBase: public Messager{
public:
    virtual void PlaySound(){
        // *****
    }
    virtual void DrawShape(){
        // *****
    }
    virtual void WriteText(){
        // *****
    }
    virtual void Connect(){
        // *****
    }
}
2.3、针对同平台不同量级的实现
  • PCMessagerLite:轻量级实现,该实现类只完成最基本的功能,普通登录、发送图片文本消息
  • PCMessagerPerfect:完美级实现,在执行这些操作之前可以辅助带一些播放声音等额外的功能
class PCMessagerLite: public PCMessagerBase{
public:
    virtual void Login(){

        PCMessagerBase::Connect();
        // ******
    }
    virtual void SendMessage(){
        PCMessagerBase::WriteText();
        // ******
    }
    virtual void SendPicture(){
        PCMessagerBase::DrawShape();
        // ******
    }
};

class PCMessagerPerfect: public PCMessagerBase{
public:
    virtual void Login(){
        PCMessagerBase::PlaySound();
        PCMessagerBase::Connect();
        // ******
    }
    virtual void SendMessage(){
        PCMessagerBase::PlaySound();
        PCMessagerBase::WriteText();
        // ******
    }
    virtual void SendPicture(){
        PCMessagerBase::PlaySound();
        PCMessagerBase::DrawShape();
        // ******
    }
};


/* 

	MobileMessagerLite、MobileMessagerPerfect的实现如上
	略过
*/
2.4、分析
  • 很明显针对上面的代码一眼就能看出:存在大量的代码重复PCMessagerLite和MobileMessagerLite的代码区别完全就在于类名不同、继承的Base类不同

    • 这一问题很好解决:首先把继承变成组合,把继承的Base类组合到Lite和Perfect类中

    • 此时只是子类不同:由于子类都来自同一个基类Messager,那么可以使用多态,进行成员声明类型的改变。

    • 最后在使用时动态的传入所需要的不同平台的实现即可。

  • 这样实现会导致一个类的急剧膨胀,假设只有1个Messager的接口、n个Base类、那么不同量级的实现是n * m的,总体是一个 1 + n + n * m + …

  • 这一点和装饰器模式很像,不过装饰器模式是一个阶乘的爆炸,但这里的工作量也不低

3、桥模式

针对上面所发现的弊端进行改进,首先可以进行的改进

3.1、改进(一)
  • 将继承改成组合的方式,可以很容易的写出PCMessagerLite、MobileMessagerLite的实现代码
class PCMessagerLite{
private:
    PCMessagerBase* pcMessagerBase;
public:
    virtual void Login(){

        pcMessagerBase->Connect();
        // ******
    }
    virtual void SendMessage(){
        pcMessagerBase->WriteText();
        // ******
    }
    virtual void SendPicture(){
        pcMessagerBase->DrawShape();
        // ******
    }
};

class MobileMessagerLite{
private:
    MobileMessagerBase* mobileMessagerBase;
public:
    virtual void Login(){
        mobileMessagerBase->Connect();
        // ******
    }
    virtual void SendMessage(){
        mobileMessagerBase->WriteText();
        // ******
    }
    virtual void SendPicture(){
        mobileMessagerBase->DrawShape();
        // ******
    }
};
3.2、改进(二)
  • 然后其实可以将PCMessagerLite、MobileMessagerLite提炼成一个代码,这两份代码核心不点在于各自做组合的对象不同
  • 然而恰巧不巧的事:它们所组合的对象都来自同一个基类,因此可以提炼成MessagerLite代码
class MessagerLite{
private:
    Messager* messager;
public:
    MessagerLite(Messager *msg): messager(msg){
        
    }
    virtual void Login(){
        messager->Connect();
        // ******
    }
    virtual void SendMessage(){
        messager->WriteText();
        // ******
    }
    virtual void SendPicture(){
        messager->DrawShape();
        // ******
    }
};
  • 但是写完这个代码会有一个问题:基类PCMessagerBase和基类MobileMessagerBase是抽象类(没有完全实现所有的纯虚函数),而抽象类是无法实例化对象的,因此这个代码是有问题的,无法过编译。

  • 仔细分析问题核心所在点:

    • 一路下来,我们只在最后不同量级的代码编写实现Messager抽象类的login、SendMessage、SendPicture三个纯虚函数
    • 之前的那些Base基类并没有实现这三个方法才导致它们依然是一个抽象类
    • 所以问题的核心点:Messager抽象类的接口方法太多,官方一点的术语:职责不够单一(违背单一职责原则)
3.3、完美改进(三)
  • 为了解决单一职责问题,可以将Messager抽象接口的拆成两个抽象接口
  • Base类继承实现一个接口
  • 不同量级实现一个接口,然后组合Base类的指针(这里动态传入Base类不同的指针)
class Messager{
public:
    virtual void Login() = 0;
    virtual void SendMessage() = 0;
    virtual void SendPicture() = 0;

    virtual ~Messager(){

    }
};

class MessagerImpl{
    virtual void PlaySound() = 0;
    virtual void DrawShape() = 0;
    virtual void WriteText() = 0;
    virtual void Connect() = 0;
};

// PC平台实现
class PCMessagerBase: public MessagerImpl{
public:
    virtual void PlaySound(){
        // *****
    }
    virtual void DrawShape(){
        // *****
    }
    virtual void WriteText(){
        // *****
    }
    virtual void Connect(){
        // *****
    }
}

// Mobile平台实现
class MobileMessagerBase: public MessagerImpl{
public:
    virtual void PlaySound(){
        // *****
    }
    virtual void DrawShape(){
        // *****
    }
    virtual void WriteText(){
        // *****
    }
    virtual void Connect(){
        // *****
    }
}


class MessagerLite: public Messager{        // 实现抽象类
private:
    MessagerImpl* messagerImpl;             // 组合对象
public:
    MessagerLite(MessagerImpl *msgi): messagerImpl(msgi){

    }
    virtual void Login(){
        messager->Connect();
        // ******
    }
    virtual void SendMessage(){
        messager->WriteText();
        // ******
    }
    virtual void SendPicture(){
        messager->DrawShape();
        // ******
    }
};



class MessagerPerfect: public Messager{     // 实现抽象类
private:
    MessagerImpl* messagerImpl;             // 组合对象
public:
    MessagerPerfect(MessagerImpl *msgi): messagerImpl(msgi){

    }
    virtual void Login(){
        MobileMessagerBase::PlaySound();
        MobileMessagerBase::Connect();
        // ******
    }
    virtual void SendMessage(){
        MobileMessagerBase::PlaySound();
        MobileMessagerBase::WriteText();
        // ******
    }
    virtual void SendPicture(){
        MobileMessagerBase::PlaySound();
        MobileMessagerBase::DrawShape();
        // ******
    }
};
  • 这样就能完成所有的功能:其核心点在于使用继承 + 组合的模式取代单一的继承方式来实现所有的功能
  • 采用桥模式的设计方式将类的膨胀改写成:1 + n + m的形式,整整比普通模式少了至少一个数量级的代码
4、总结
  • Bridge模式使用“对象间的组合关系”解耦了抽象和实现之间固有的绑定关系,使得抽象和实现可以沿着各自的维度来变化。所谓抽象和实现沿着各自维度的变化,即“子类化”它们。
  • Bridge模式有时候类似于多继承方案,但是多继承方案往往违背单一职责原则(即一个类存在多个变化的原因和方向),复用性比较差。Bridge模式是比多级车工方案更好的解决方案。
  • Bridge模式的应用一般在“两个非常强的变化维度”,有时一个类也可以有多余两个的变化维度,这时可以使用Bridge的扩展模式。
    在这里插入图片描述

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

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

相关文章

Jenkins 持续集成 【CICD】

持续集成 (Continuous integration,简称CI) 持续集成是一种开发实践,它倡导团队成员频繁的集成他们的工作,每次集成都通过自动化构建(包括编译、构建、打包、部署、自动化测试)来验证&#xff…

libVLC 提取视频帧

在前面的文章中,我们使用libvlc_media_player_set_hwnd设置了视频的显示的窗口。 libvlc_media_player_set_hwnd(vlc_mediaPlayer, (void *)ui.widgetShow->winId()); 如果我们想要提取每一帧数据,将数据保存到本地,该如何操作呢&#x…

思迈特软件与上海德拓签署战略合作协议,携手赋能企业数字化转型

3月27日,广州思迈特软件有限公司(简称“思迈特软件”)与上海德拓信息技术有限公司(简称“德拓信息”)正式签约建立战略合作伙伴关系。双方将在数字化转型、数据服务、数据应用以及市场资源等多个领域展开深度合作&…

Github上传大文件(>25MB)教程

0.在github中创建新的项目(已创建可忽略这一步) 如上图所示,点击New repository 进入如下页面: 1.下载Git LFS 下载git 2.打开gitbash 3.上传文件,代码如下: cd upload #进入名为upload的文件夹,提前…

距离度量方法——欧氏距离、曼哈顿距离、切比雪夫距离、闵可夫斯基距离

目录 一、 欧氏距离(Euclidean Distance) 1、简介 2、代码实现 二、曼哈顿距离(Manhattan Distance) 1、简介 2、代码实现 三、切比雪夫距离(Chebyshev Distance) 1、简介 2、代码实现 四、闵可夫…

element-ui 在Popover弹框中使用Select选择器,Vue3

bug描述: 当选择完select的时候,popover也会退出。 解决: popover组件的的关闭是当点击组件外的元素时会关闭,select虽然是写在组件内的,但是select有一个默认属性teleported“true” 会把它默认插到 body 元素,我…

C++ //练习 11.22 给定一个map<string, vector<int>>,对此容器的插入一个元素的insert版本,写出其参数类型和返回类型。

C Primer&#xff08;第5版&#xff09; 练习 11.22 练习 11.22 给定一个map<string, vector<int>>&#xff0c;对此容器的插入一个元素的insert版本&#xff0c;写出其参数类型和返回类型。 环境&#xff1a;Linux Ubuntu&#xff08;云服务器&#xff09; 工具…

MySQL高级篇(B-Tree、Btree)

目录 1、Btree&#xff08;B-Tree&#xff09; 1.1、B-Trees的特点 二叉树缺点&#xff1a;顺序插入时&#xff0c;会形成一个链表&#xff0c;查询性能大大降低。大数据量情况下&#xff0c;层级较深&#xff0c;检索速度慢。红黑树&#xff1a;大数据量情况下&#xff0c;层…

Qt+VS2019中使用QAxObject时的环境配置

在纯Qt中 在.pro中添加axcontainer模块即可 而VSqt中&#xff1a; 特别傻的是&#xff1a;我运行的是release&#xff0c;但配置的是debug的属性页&#xff0c;一直报错&#xff0c;人都傻了。 最后发现果然是人傻。

如何远程监控员工的电脑

如何远程监控员工的电脑 为什么要对员工使用电脑的行为进行监控呢&#xff1f;对企业来说是有其必要性的。其必要性主要体现以下4个方面。 1.数据安全与知识产权保护 防止数据泄露&#xff1a;企业内部往往存储着大量的敏感信息&#xff0c;如客户数据、财务资料、商业秘密、…

推荐一款很强大的SCADA工业组态软件

可以广泛应用于化工、石化、制药、冶金、建材、市政、环保、电力等几十个行业。 I官网网站:www.hcy-soft.com |体验地址:http://www.byzt.net:60/sm/ 一、产品简介 BY组态是完全自主研发的集实时数据展示、动态交互等一体的全功能可视化平台。帮助物联网、工业互联网、电力能…

Hyper-v平台搭建pve系统之网络配置(双网卡、内外网分离)

现在我需要在我本地配置的PVE系统上配置双网卡&#xff0c;然后一个连接外部网络&#xff08;访问互联网&#xff09;&#xff0c;一个连接内部网络&#xff08;只能和宿主机之间互相访问&#xff09; 最终效果&#xff1a; 登录PVE平台&#xff0c;我可以正常访问外网&#…

信息工程大学第五届超越杯程序设计竞赛(同步赛)vp

目录 信息工程大学第五届超越杯程序设计竞赛&#xff08;同步赛&#xff09;_ACM/NOI/CSP/CCPC/ICPC算法编程高难度练习赛_牛客竞赛OJ A.遗失的旋律 C.财政大臣 D.实验室有多少人 E.在雾中寻宁静 F.不规则的轮回 G.完美数字 M.Monikas game A.遗失的旋律 思路&#xf…

Golang 开发实战day09 - package Scope

&#x1f3c6;个人专栏 &#x1f93a; leetcode &#x1f9d7; Leetcode Prime &#x1f3c7; Golang20天教程 &#x1f6b4;‍♂️ Java问题收集园地 &#x1f334; 成长感悟 欢迎大家观看&#xff0c;不执着于追求顶峰&#xff0c;只享受探索过程 Golang 教程09 - package Sc…

大学英语ab级题搜题软件?分享7个支持答案和解析的工具 #笔记#其他

合理利用学习辅助工具和资料&#xff0c;可以帮助大学生更好地组织学习内容、掌握知识点和提升学术水平。 1.智能翻译官 这是一款多语言在线翻译神器&#xff0c;除了最基础的英语以外&#xff0c;还支持日语、德语、俄语、法语等几十种语言文本翻译和拍照翻译&#xff0c;并…

Your file appears not to be a valid OLE2 document

前言 org.apache.poi.poifs.filesystem.NotOLE2FileException:Invalid header signature; read 0x0000000000000000, expected 0xE11AB1A1E011CFD0 - Your file appears not to be a valid OLE2 document解决 Excel兼容模式打开老版本文件造成文件损坏&#xff0c;大多说的是点…

期货量化交易软件:模式搜索的暴力方法(第六部分)循环优化

概述 考虑到我上一篇文章中的材料&#xff0c;我赫兹量化软件可以说这只是我在算法中引入的所有函数的肤浅描述。它们不仅涉及EA创建的完全自动化&#xff0c;还涉及诸如结果优化和选择的完全自动化以及随后用于自动交易&#xff0c;或者我赫兹量化稍后将展示的更先进的EA的创…

关于Ansible模块 ⑤

转载说明&#xff1a;如果您喜欢这篇文章并打算转载它&#xff0c;请私信作者取得授权。感谢您喜爱本文&#xff0c;请文明转载&#xff0c;谢谢。 继《关于Ansible的模块 ①》、《关于Ansible的模块 ②》与《关于Ansible的模块 ③》之后&#xff0c;继续学习ansible常用模块之…

高效测试丨怿星RTP协议测试解决方案

近几年&#xff0c;车内音视频娱乐系统不断发展&#xff0c;功能不断丰富&#xff0c;对于音视频的传输需求也逐渐增多&#xff0c;随着车载以太网的日渐成熟&#xff0c;各主机厂逐步方案落地、成本逐步降低&#xff0c;基于车载以太网的音视频传输也在逐步应用&#xff0c;常…

python代码使用过程中使用快捷键注释时报错

1.代码 2.代码报错 3.代码注释后的结果 4. 原因