【GeekBand】C++设计模式笔记7_Bridge_桥接模式

1. "单一职责"模式

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

2. Bridge 桥接模式

2.1 动机(Motivation)

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

2.2 模式定义

抽象 部分(业务功能)与实现 部分(平台实现)分离,使它们都可以独立地变化。

------《设计模式》GoF

2.3 实例代码

2.3.1 bridge1
cpp 复制代码
// 消息抽象基类
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() {}
};


/******************** 平台实现 ********************/
// PC端消息基类,因为没有对继承来的Login、SendMessage、SendPicture方法进行override,所以还是抽象类
class PCMessagerBase : public Messager {
public:
    virtual void PlaySound() {
        //**********
    }
    virtual void DrawShape() {
        //**********
    }
    virtual void WriteText() {
        //**********
    }
    virtual void Connect() {
        //**********
    }
};

// 移动端消息基类,因为没有对继承来的Login、SendMessage、SendPicture方法进行override,所以还是抽象类
class MobileMessagerBase : public Messager {
public:
    virtual void PlaySound() {
        //==========
    }
    virtual void DrawShape() {
        //==========
    }
    virtual void WriteText() {
        //==========
    }
    virtual void Connect() {
        //==========
    }
};

/******************** 业务抽象 ********************/
// PC端精简消息类
class PCMessagerLite : public PCMessagerBase {
public:
    virtual void Login(string username, string password) {
        
        PCMessagerBase::Connect();
        //........
    }
    virtual void SendMessage(string message) {
        
        PCMessagerBase::WriteText();
        //........
    }
    virtual void SendPicture(Image image) {
        
        PCMessagerBase::DrawShape();
        //........
    }
};

// PC端完全消息类
class PCMessagerPerfect : public PCMessagerBase {
public:
    virtual void Login(string username, string password) {
        
        PCMessagerBase::PlaySound();
        //********
        PCMessagerBase::Connect();
        //........
    }
    virtual void SendMessage(string message) {
        
        PCMessagerBase::PlaySound();
        //********
        PCMessagerBase::WriteText();
        //........
    }
    virtual void SendPicture(Image image) {
        
        PCMessagerBase::PlaySound();
        //********
        PCMessagerBase::DrawShape();
        //........
    }
};

// 移动端精简消息类
class MobileMessagerLite : public MobileMessagerBase {
public:
    virtual void Login(string username, string password) {
        
        MobileMessagerBase::Connect();
        //........
    }
    virtual void SendMessage(string message) {
        
        MobileMessagerBase::WriteText();
        //........
    }
    virtual void SendPicture(Image image) {
        
        MobileMessagerBase::DrawShape();
        //........
    }
};

// 移动端完全消息类
class MobileMessagerPerfect : public MobileMessagerBase {
public:
    virtual void Login(string username, string password) {
        
        MobileMessagerBase::PlaySound();
        //********
        MobileMessagerBase::Connect();
        //........
    }
    virtual void SendMessage(string message) {
        
        MobileMessagerBase::PlaySound();
        //********
        MobileMessagerBase::WriteText();
        //........
    }
    virtual void SendPicture(Image image) {
        
        MobileMessagerBase::PlaySound();
        //********
        MobileMessagerBase::DrawShape();
        //........
    }
};


void Process() {
        // 编译时装配
        Messager *m = new MobileMessagerPerfect();
}
2.3.2 bridge2
cpp 复制代码
// 消息抽象基类
class Messager {
protected:
     MessagerImp* messagerImp;	//... 组合形式,运行时动态绑定
public:
    virtual void Login(string username, string password) = 0;
    virtual void SendMessage(string message) = 0;
    virtual void SendPicture(Image image) = 0;
    
    virtual ~Messager() {}
};

/******************** 平台实现 n ********************/
// 实现抽象基类
class MessagerImp {
public:
    virtual void PlaySound() = 0;
    virtual void DrawShape() = 0;
    virtual void WriteText() = 0;
    virtual void Connect() = 0;
    
    virtual ~MessagerImp() {}
};

// PC端消息类
class PCMessagerImp : public MessagerImp {
public:
    virtual void PlaySound() {
        //**********
    }
    virtual void DrawShape() {
        //**********
    }
    virtual void WriteText() {
        //**********
    }
    virtual void Connect() {
        //**********
    }
};

// 移动端消息类
class MobileMessagerImp : public MessagerImp {
public:
    virtual void PlaySound() {
        //==========
    }
    virtual void DrawShape() {
        //==========
    }
    virtual void WriteText() {
        //==========
    }
    virtual void Connect() {
        //==========
    }
};
/******************** 业务抽象 m ********************/
//类的数目:1+n+m

// 精简消息类
class MessagerLite : public Messager {   
public:
	// 构造函数
	MessagerLite(MessagerImp* mImp) : messagerImp(mImp) {}
	
    virtual void Login(string username, string password) {
        
        messagerImp->Connect();
        //........
    }
    virtual void SendMessage(string message) {
        
        messagerImp->WriteText();
        //........
    }
    virtual void SendPicture(Image image) {
        
        messagerImp->DrawShape();
        //........
    }
};

// 完全消息类
class MessagerPerfect : public Messager {
public:
	// 构造函数 
	MessagerPerfect(MessagerImp* mImp) : messagerImp(mImp) {}
	 
    virtual void Login(string username, string password) {
        
        messagerImp->PlaySound();
        //********
        messagerImp->Connect();
        //........
    }
    virtual void SendMessage(string message) {
        
        messagerImp->PlaySound();
        //********
        messagerImp->WriteText();
        //........
    }
    virtual void SendPicture(Image image) {
        
        messagerImp->PlaySound();
        //********
        messagerImp->DrawShape();
        //........
    }
};

void Process() {
    // 运行时装配
    MessagerImp* mImp = new PCMessagerImp();
    Messager *m = new MessagerLite(mImp);	// m中的messagerImp具体类型在运行时动态装配
}

2.4 结构(Structure)

2.5 要点总结

  • Bridge 模式使用 "对象间的组合关系" 解耦了抽象 (PS:实例代码中的 Messager 类)和实现(PS:实例代码中的 MessagerImp 类)之间固有的绑定关系,使得抽象和实现可以沿着各自的维度来变化。所谓抽象和实现沿着各自维度的变化,即 "子类化" (PS:实例代码中的 MessagerLite / MessagePerfect 和 PCMessagerImp / MobileMessagerImp)它们。
  • Bridge 模式有时候类似于多继承方案,但是多继承方案往往违背单一职责原则(即一个类只有一个变化的原因),复用性比较差。Bridge 模式是比多继承方案更好的解决方法。
  • Bridge 模式的应用一般在 "两个非常强的变化维度",有时一个类也有多余两个的变化维度,这时可以使用 Bridge 的扩展模式。
相关推荐
zhooyu14 分钟前
C++和OpenGL实现3D游戏编程【连载14】——VBO、VAO和EBO应用
开发语言·c++·游戏·游戏程序·游戏策划
anyup_前端梦工厂16 分钟前
JS设计模式之职责链模式:优雅地处理请求流程
前端·javascript·设计模式·责任链模式
QuantumStack20 分钟前
【C++ 真题】B2003 输出第二个整数
开发语言·c++·算法
无敌开心1 小时前
Web前端高级工程师培训:设计模式相关
前端·设计模式
是jin奥1 小时前
C++ inline 的更进一步理解
开发语言·c++
咕噜Yuki06091 小时前
Python和C++的差异在哪里
开发语言·c++·python
伤心男孩拯救世界(Code King)1 小时前
【优选算法】--- 位运算
c++·算法
594h22 小时前
传智杯 第六届—C
数据结构·c++·算法
single5942 小时前
【优选算法】(第三十四篇)
java·开发语言·数据结构·c++·python·算法·leetcode
破晓的历程2 小时前
C++设计模式——代理模式
c++·设计模式·代理模式