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 的扩展模式。