C++设计模式之桥模式
创始人
2024-04-10 18:46:48
0

桥模式也是设计模式中单一组件模式的一种。什么是单一组件模式呢?

单一组件模式:

  • 在软件组件设计中,如果责任划分的不清晰,使用继承得到的结果往往是随着需求的变化而变化,子类急剧膨胀,同时充斥着重复代码,这时候的关键是划清责任。 
  • 典型的单一组件模式有装饰模式和桥模式。

如下代码是我们没有学习过桥模式写出的代码,下面代码通过功能分解的方法将一个复杂功能伪代码实现。下面代码挺简单,我就不仔细说明了。

class Messager {
public:virtual void Login(string username, string password) = 0;virtual void SendMessage(string message) = 0;virtual void SendPicture(Image image) = 0;virtual void PlaySound() = 0;virtual void DrawShape() = 0;virtual void WriteText() = 0;virtual void Connect() = 0;virtual ~Messager(){}
};// 平台实现
class PCMessagerbase :public Messager { //PC平台
public:virtual void PlaySound() {//**********}virtual void PlaySound() {//**********}virtual void WriteText() {//**********}virtual void Connect() {//**********}
};
class MobileMessagerbase :public Messager { //移动平台
public:virtual void PlaySound() {//**********}virtual void PlaySound() {//**********}virtual void WriteText() {//**********}virtual void Connect() {//**********}
};
// 业务抽象
class PCMessageLite : public PCMessagerbase {
public:virtual void Login(string username, string password) {PCMessagerbase::Connect();//**********}virtual void SendMassage(string message) {PCMessagerbase::WriteText();//*********}virtual void SendPicture(Image image) {PCMessagerbase::DrawShape();//*********}
};
class PCMessagePerfect : public PCMessagerbase {
public:virtual void Login(string username, string password) {PCMessagerbase::PlaySound();PCMessagerbase::Connect();//**********}virtual void SendMassage(string message) {PCMessagerbase::WriteText();//*********PCMessagerbase::PlaySound();}virtual void SendPicture(Image image) {PCMessagerbase::PlaySound();PCMessagerbase::DrawShape();//*********}
};
class MobileMessageLite : public MobileMessagerbase {
public:virtual void Login(string username, string password) {MobileMessagerbase::Connect();//**********}virtual void SendMassage(string message) {MobileMessagerbase::WriteText();//*********}virtual void SendPicture(Image image) {MobileMessagerbase::DrawShape();//*********}
};
class MobileMessagePerfect : public MobileMessagerbase {
public:virtual void Login(string username, string password) {MobileMessagerbase::PlaySound();MobileMessagerbase::Connect();//**********}virtual void SendMassage(string message) {MobileMessagerbase::WriteText();//*********MobileMessagerbase::PlaySound();}virtual void SendPicture(Image image) {MobileMessagerbase::PlaySound();MobileMessagerbase::DrawShape();//*********}
};void process() {//编译时装配Messager* m = new MobileMessagePerfect();
}

将上述代码如何修改为桥模式呢?为什么能够修改为桥模式呢?

动机:

  • 由于某些类型的固有实现逻辑,使得他们具有两个变化的维度,乃至多个维度的变化。
  • 如何应对多维度的变化?如何利用面向对象技术使得类型可以轻松地沿着两个乃至多个维度变化,而不引入额外的复杂度?

从上述子问题拆分方式进行业务代码实现,我们实现了很多类,基类1个,平台实现类2个,业务抽象类2*2个。然而在实际业务中,你的平台实现和业务抽象类将会更多,假设n个实现平台,m个业务抽象方法,那将会有1+n+m*n个类,类的个数随着平台和业务抽象急剧膨胀!

        下面代码是修改后代码,用桥模式的实现上述业务。基类Messager分为两个类,一个是Messager类,一个是MessagerImp类,平台的选择通过多态指针messagerImp实现,在运行时被确定。下面的代码将平台变化和业务变化两个维度进行,带动了行为的多态实现,这也就导致未修改前的基类Messager要分为两个类。实现了将抽象部分(业务功能)与实现部分(平台实现)分离,是他们可以独立的变化

        

class Messager { //第一个变化方向,平台变化
public:Messager(MessagerImp* imp):messagerImp(imp){} //初始化多态指针MessagerImp* messagerImp; //多态指针决定未来运行时是那个平台virtual void Login(string username, string password) = 0;virtual void SendMessage(string message) = 0;virtual void SendPicture(Image image) = 0;virtual ~Messager() {}
};
class MessagerImp { 
public:virtual void PlaySound() = 0;virtual void DrawShape() = 0;virtual void WriteText() = 0;virtual void Connect() = 0;virtual ~MessagerImp() {};
};// 平台实现
class PCMessagerImp :public MessagerImp { //PC平台
public:virtual void PlaySound() {//**********}virtual void PlaySound() {//**********}virtual void WriteText() {//**********}virtual void Connect() {//**********}
};
class MobileMessagerImp :public MessagerImp { //移动平台
public:virtual void PlaySound() {//**********}virtual void PlaySound() {//**********}virtual void WriteText() {//**********}virtual void Connect() {//**********}
};
// 业务抽象
//第二个变化,业务的变化(简洁版和完美版)
class MessageLite:public Messager {public:virtual void Login(string username, string password) {messagerImp->Connect();//**********}virtual void SendMassage(string message) {messagerImp->WriteText();//*********}virtual void SendPicture(Image image) {messagerImp->DrawShape();//*********}
};
class MessagePerfect : public Messager{
public:virtual void Login(string username, string password) {messagerImp->PlaySound();messagerImp->Connect();//**********}virtual void SendMassage(string message) {messagerImp->WriteText();//*********messagerImp->PlaySound();}virtual void SendPicture(Image image) {messagerImp->PlaySound();messagerImp->DrawShape();//*********}
};void process() {//运行时装配MessagerImp* mImp = new PCMessagerImp();Messager* m = new Messager(mImp);
}

如下图所示是上述代码的桥模式类图

 总结:

  • Bridge模式使用“对象间的组合关系"解耦了抽象和实现之间固有的绑定关系,使得抽象和实现可以沿着各自的维度来变化。所谓抽象和实现沿着各自纬度的变化,即“子类化”它们。
  • Bridge模式有时候类似于多继承方案,但是多继承方案往往违背单一职责原则(即一个类只有一个变化的原因),复用性比较差。Bridge模式是比多继承方案更好的解决方法。
  • Bridge模式的应用一般在“两个非常强的变化维度”,有时一个类也有多于两个的变化维度,这时可以使用Bridge的扩展模式。

相关内容

热门资讯

不能访问光猫的的管理页面 光猫是现代家庭宽带网络的重要组成部分,它可以提供高速稳定的网络连接。但是,有时候我们会遇到不能访问光...
【NI Multisim 14...   目录 序言 一、工具栏 🍊1.“标准”工具栏 🍊 2.视图工具...
Android|无法访问或保存... 这个问题可能是由于权限设置不正确导致的。您需要在应用程序清单文件中添加以下代码来请求适当的权限:此外...
银河麒麟V10SP1高级服务器... 银河麒麟高级服务器操作系统简介: 银河麒麟高级服务器操作系统V10是针对企业级关键业务...
北信源内网安全管理卸载 北信源内网安全管理是一款网络安全管理软件,主要用于保护内网安全。在日常使用过程中,卸载该软件是一种常...
安卓文字转语音tts没有声音 安卓文字转语音TTS没有声音的问题在应用中比较常见,通常是由于一些设置或者代码逻辑问题导致的。本文将...
APK正在安装,但应用程序列表... 这个问题可能是由于以下原因导致的:应用程序安装的APK文件可能存在问题。设备上已经存在同名的应用程序...
​ToDesk 远程工具安装及... 目录 前言 ToDesk 优势 ToDesk 下载安装 ToDesk 功能展示 文件传输 设备链接 ...
AWSECS:访问外部网络时出... 如果您在AWS ECS中部署了应用程序,并且该应用程序需要访问外部网络,但是无法正常访问,可能是因为...
报告实验.pdfbase.tt... 这个错误通常是由于找不到字体文件或者文件路径不正确导致的。以下是一些解决方法:确认字体文件是否存在:...