一、概念
将一个类的接口转换成客户希望的另一个接口。Adapter模式使得原本由于接口不兼容而不能一起工作的那些类可以一起工作。—《设计模式》Gof
二、动机
在软件系统中,由于应用环境的变化,常常需要将“一些现存的对象”放在新的环境中应用,但是新环境要求的接口是这些现存对象所不满足的。
如何应对这种“迁移的变化”?如何既能利用现有对象的良好实现,同时又能满足新的应用环境所要求的接口?
三、适配器模式的实现
// 目标接口(新接口)
class ITarget {
public:
virtual void process() = 0;
};
// 遗留接口(老接口)
class IAdaptee {
public:
virtual void foo(int data) = 0;
virtual int bar() = 0;
};
// 遗留类型,符合老的接口
class OldClass : public IAdaptee {
// ...
};
// 对象适配器
class Adapter: public ITarget { // 继承
protected:
IAdaptee *pAdaptee; // 组合
public:
Adapter(IAdaptee *pAdaptee) {
this->pAdaptee = pAdaptee;
}
virtual void process() {
int data = pAdaptee->bar();
pAdaptee->foo(data);
}
};
int main() {
IAdaptee *pAdaptee = new OldClass();
ITarget *pTarget = new Adapter(pAdaptee);
pTarget->process();
}
四、类图结构
五、要点总结
-
Adapter模式主要应用于“希望复用一些现存的类,但是接口又与服用环境要求不一致的情况”,在遗留代码复用、类库迁移等方面非常有用。
-
GoF23定义了两种Adapter模式的结构实现:对象适配器和类适配器。但类适配器采用“多继承”的实现方式,一般不推荐使用。对象适配器采用“对象组合”的方式,更符合松耦合的精神。
-
Adapter模式可以实现的非常灵活,不必拘泥于GoF23中定义的两种结构。例如,完全可以将Adapter模式中的“现存对象”作为新的接口方法参数,来达到适配的目的。