C++设计模式之桥模式
创始人
2024-02-03 19:29:00
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的扩展模式。

相关内容

热门资讯

喜欢穿一身黑的男生性格(喜欢穿... 今天百科达人给各位分享喜欢穿一身黑的男生性格的知识,其中也会对喜欢穿一身黑衣服的男人人好相处吗进行解...
发春是什么意思(思春和发春是什... 本篇文章极速百科给大家谈谈发春是什么意思,以及思春和发春是什么意思对应的知识点,希望对各位有所帮助,...
网络用语zl是什么意思(zl是... 今天给各位分享网络用语zl是什么意思的知识,其中也会对zl是啥意思是什么网络用语进行解释,如果能碰巧...
为什么酷狗音乐自己唱的歌不能下... 本篇文章极速百科小编给大家谈谈为什么酷狗音乐自己唱的歌不能下载到本地?,以及为什么酷狗下载的歌曲不是...
华为下载未安装的文件去哪找(华... 今天百科达人给各位分享华为下载未安装的文件去哪找的知识,其中也会对华为下载未安装的文件去哪找到进行解...
怎么往应用助手里添加应用(应用... 今天百科达人给各位分享怎么往应用助手里添加应用的知识,其中也会对应用助手怎么添加微信进行解释,如果能...
家里可以做假山养金鱼吗(假山能... 今天百科达人给各位分享家里可以做假山养金鱼吗的知识,其中也会对假山能放鱼缸里吗进行解释,如果能碰巧解...
四分五裂是什么生肖什么动物(四... 本篇文章极速百科小编给大家谈谈四分五裂是什么生肖什么动物,以及四分五裂打一生肖是什么对应的知识点,希...
一帆风顺二龙腾飞三阳开泰祝福语... 本篇文章极速百科给大家谈谈一帆风顺二龙腾飞三阳开泰祝福语,以及一帆风顺二龙腾飞三阳开泰祝福语结婚对应...
美团联名卡审核成功待激活(美团... 今天百科达人给各位分享美团联名卡审核成功待激活的知识,其中也会对美团联名卡审核未通过进行解释,如果能...