• Adapter 适配器


    意图

    将一个类的接口转换成客户希望的另外一个接口。Adapter模式使得原本由于接口不兼容而不能一起工作的那些类可以一起工作。

    动机

    在软件开发中,有的时候系统的数据和行为都正确,但接口不符合,我们应该考虑使用适配器模式,目的是使控制范围之外的一个原有对象与某个接口匹配。举个例子:在开发一个模块的时候,有一个功能点实现起来比较费劲,但是,之前有一个项目的模块实现了一样的功能点;但是现在这个模块的接口和之前的那个模块的接口是不一致的。此时,作为项目经理的你,该怎么办啦?当然是在中间加一层Wrapper了,也就是使用适配器模式,将之前实现的功能点适配进新的项目了。为什么呢?主要是使用适配器模式有以下优点:

    1. 降低了去实现一个功能点的难度,可以对现有的类进行包装,就可以进行使用了;
    2. 提高了项目质量,现有的类一般都是经过测试的,使用了适配器模式之后,不需要对旧的类进行全面的覆盖测试;
    3. 总的来说,提高了效率,降低了成本。

    适用性

    1. 使用一个已经存在的类,如果它的接口和你实际要求的不一致时,可以考虑使用适配器模式;
    2. 要在调用者和功能提供者双方都不太容易修改的时候再使用适配器模式,而不是一有不同时就使用它。

    结构

    类适配器:

    上图是适配器模式的第一种实现形式,适配器Adapter继承自Target和Adaptee类,Adapter类需要重写Target类的Request函数,在Request中做适当的处理,调用Adaptee类的SepcificRequest。最终,Target实际调用的是Adaptee的SpecificRequest来完成Request的,完成适配;这种叫做类适配器。

    对象适配器:

    上图是适配器的第二种实现形式,适配器Adapter类继承自Target类,同时,在Adapter类中有一个Adaptee类型的成员变量;Adapter类重写Request函数时,在Request中,使用Adaptee类型的成员变量调用Adaptee的SpecificRequest函数,最终完成适配;这种叫做对象适配器。

    类适配器有以下特点:

    1. 由于Adapter直接继承自Adaptee类,所以,在Adapter类中可以对Adaptee类的方法进行重定义;
    2. 如果在Adaptee中添加了一个抽象方法,那么Adapter也要进行相应的改动,这样就带来高耦合;
    3. 如果Adaptee还有其它子类,而在Adapter中想调用Adaptee其它子类的方法时,使用类适配器是无法做到的。

    对象适配器有以下特点:

    1. 有的时候,你会发现,不是很容易去构造一个Adaptee类型的对象;
    2. 当Adaptee中添加新的抽象方法时,Adapter类不需要做任何调整,也能正确的进行动作;
    3. 可以使用多肽的方式在Adapter类中调用Adaptee类子类的方法。

    由于对象适配器的耦合度比较低,所以在很多的书中都建议使用对象适配器。在我们实际项目中,也是如此,能使用对象组合的方式,就不使用多继承的方式。

    实现

    对象适配器

     1 #include <iostream>
     2 using namespace std;
     3 
     4 class Target
     5 {
     6 public:
     7     Target(){}
     8     virtual ~Target(){}
     9     virtual void Request()
    10     {
    11         cout<<"Target::Request"<<endl;
    12     }
    13 };
    14 
    15 class Adaptee
    16 {
    17 public:
    18     void SpecificRequest()
    19     {
    20         cout<<"Adaptee::SpecificRequest"<<endl;
    21     }
    22 };
    23 
    24 class Adapter : public Target
    25 {
    26 public:
    27     Adapter() : m_Adaptee(new Adaptee) {}
    28 
    29     ~Adapter()
    30     {
    31         if (m_Adaptee != NULL)
    32         {
    33             delete m_Adaptee;
    34             m_Adaptee = NULL;
    35         }
    36     }
    37 
    38     void Request()
    39     {
    40         m_Adaptee->SpecificRequest();
    41     }
    42 
    43 private:
    44     Adaptee *m_Adaptee;
    45 };
    46 
    47 int main(int argc, char *argv[])
    48 {
    49     Target *targetObj = new Adapter();
    50     targetObj->Request();
    51 
    52     delete targetObj;
    53     targetObj = NULL;
    54 
    55     return 0;
    56 }

    类适配器

     1 #include <iostream>
     2 using namespace std;
     3 
     4 // Targets
     5 class Target
     6 {
     7 public:
     8     virtual void Request()
     9     {
    10         cout<<"Target::Request"<<endl;
    11     }
    12 };
    13 
    14 // Adaptee
    15 class Adaptee
    16 {
    17 public:
    18     void SpecificRequest()
    19     {
    20         cout<<"Adaptee::SpecificRequest"<<endl;
    21     }
    22 };
    23 
    24 // Adapter
    25 class Adapter : public Target, Adaptee
    26 {
    27 public:
    28     void Request()
    29     {
    30         Adaptee::SpecificRequest();
    31     }
    32 };
    33 
    34 // Client
    35 int main(int argc, char *argv[])
    36 {
    37     Target *targetObj = new Adapter();
    38     targetObj->Request();
    39 
    40     delete targetObj;
    41     targetObj = NULL;
    42 
    43     return 0;
    44 }
  • 相关阅读:
    Python学习之路(一)之Python基础1
    【第十一课】Tomcat原理解析【转】
    【第十课】Tomcat入门
    【第九课】MriaDB密码重置和慢查询日志
    【第八课】php-fpm.conf配置文件解析
    【第七课】Nginx反向代理和负载均衡
    【第六课】Nginx常用配置下详解
    maven项目中更新了核心库后导致一些包提示未定义,如:The import org.json cannot be resolved
    js代码突然在花括号回车自动多加了一个大括号
    SQL0419N 十进制除法运算无效,因为结果将有一个负小数位。 SQLSTATE=42911
  • 原文地址:https://www.cnblogs.com/raichen/p/5676702.html
Copyright © 2020-2023  润新知