【设计模式】封装器模式:适配器模式

意图

适配器模式能使接口不兼容的对象能够相互合作。

让汽车能够在铁轨上行驶

问题

假如你正在使用 C++ 开发一个程序,在开发过程中需要使用一系列用 C 语言编写的函数,需要将一个类传入到这些函数中。而 C 语言是不支持类这一概念的,C 语言实现类,只能通过用结构体模拟类。
为了用户的使用体验,你还是希望用户使用的是C++中的类,而不是使用结构体模拟出来的类。但是如果将那一系列 C 函数重新用 C++ 实现一遍,工作量就会非常大。


image

解决方案

你可以创建一个适配器,这是一个特殊的对象,能够转换对象接口,使其能与其他对象进行交互。适配器模式通过封装对象将复杂的转换过程隐藏于幕后,被封装的对象甚至察觉不到适配器的存在。
在刚才的问题中就可以如下图,在类和 C 函数中间加一个适配器,通过转换函数,将类转换成结构体。由于适配器继承自结构体,所以可以将适配器作为结构体直接传入 C 函数中。而需要调用结构体方法时,则通过 m_class 这个成员变量访问类的函数。


image

示例代码

以下是 AWTK-MVVM 使用适配器的部分示例代码:

/* temperature_validator.cpp */
#include "temperature_validator.hpp"
#include "mvvm/cpp/adapter.hpp"

class TemperatureValidator : public vm::ValueValidator {
 public:
  virtual bool_t IsValid(const value_t* value, str_t* msg) {
    int32_t temp = value_int(value);

    if (temp <= 20) {
      str_set(msg, "too low");
      return FALSE;
    } else if (temp >= 60) {
      str_set(msg, "too high");
      return FALSE;
    } else {
      str_set(msg, "normal");
      return TRUE;
    }
  }

  virtual ret_t Fix(value_t* value) {
    return RET_OK;
  }
};

static void* create_water_temp_validator(void) {
  /* class 转换为 struct */
  return vm::To(new TemperatureValidator());
}

ret_t temperature_validator_init(void) {
  value_validator_register("water_temp", create_water_temp_validator);

  return RET_OK;
}
/* value_validator.c */
/* C函数 */
ret_t value_validator_register(const char* name, tk_create_t create) {
  return_value_if_fail(name != NULL, RET_BAD_PARAMS);
  return_value_if_fail(create != NULL && s_validator_factory != NULL, RET_BAD_PARAMS);

  return object_set_prop_pointer(s_validator_factory->creators, name, create);
}
/* value_validator.h */
/**
 * @class value_validator_t
 * @parent object_t
 *
 * 值校验器。
 *
 * 用户在界面上输入的类型可能是无效的,value_validator负责将检查用户输入的有效性。
 *
 */
struct _value_validator_t {
  object_t object;

  object_t* context;

  /*private*/
  value_validator_is_valid_t is_valid;
  value_validator_fix_t fix;
}value_validator_t;
/* adapter.cpp */
namespace vm {
static object_vtable_t s_value_validator_adapter_vtable;

typedef struct _value_validator_adapter_t {
  /* 继承: value_validator_adapter_t -> value_validator_t -> object_t */
  value_validator_t value_validator;

  /*private*/
  ValueValidator* cpp;
} value_validator_adapter_t;

static ret_t value_validator_adapter_init_vtable(object_vtable_t* vt) {
  vt->type = "value_validator_adapter";
  vt->desc = "value_validator_adapter";
  vt->size = sizeof(value_validator_adapter_t);
  vt->is_collection = FALSE;

  return RET_OK;
}

#define VALUE_VALIDATOR_ADAPTER(obj) ((value_validator_adapter_t*)(obj))

static bool_t value_validator_adapter_is_valid(value_validator_t* c, const value_t* value,
                                               str_t* msg) {
  value_validator_adapter_t* value_convert_adapter = VALUE_VALIDATOR_ADAPTER(c);

  /* 调用 class 函数 */
  return value_convert_adapter->cpp->IsValid(value, msg);
}

static ret_t value_validator_adapter_fix(value_validator_t* c, value_t* value) {
  value_validator_adapter_t* value_convert_adapter = VALUE_VALIDATOR_ADAPTER(c);

  return value_convert_adapter->cpp->Fix(value);
}

value_validator_t* value_validator_cpp_create(ValueValidator* cpp) {
  object_t* obj = NULL;
  value_validator_t* value_convert = NULL;
  return_value_if_fail(cpp != NULL, NULL);
  value_validator_adapter_t* value_convert_adapter = NULL;

  value_validator_adapter_init_vtable(&s_value_validator_adapter_vtable);
  obj = object_create(&s_value_validator_adapter_vtable);
  return_value_if_fail(obj != NULL, NULL);

  value_convert = VALUE_VALIDATOR(obj);
  value_convert->fix = value_validator_adapter_fix;
  value_convert->is_valid = value_validator_adapter_is_valid;

  value_convert_adapter = VALUE_VALIDATOR_ADAPTER(obj);
  value_convert_adapter->cpp = cpp;

  return value_convert;
}

value_validator_t* To(ValueValidator* cpp) {
  return value_validator_cpp_create(cpp);
}
}  // namespace vm

适配器模式结构

对象适配器

使用了构成原则: 适配器实现了其中一个对象的接口, 并对另一个对象进行封装。
上述示例代码用的就是对象适配器。

特点

  • 当 Service 类较为复杂时,构造一个 Service 类对象比较困难。
  • 在 Service 类新增抽象方法,Adapter 类无需修改,也能正常使用。
  • 可以使用多态的方式在 Adapter 类中调用 Service 子类的方法。


    对象适配器URL图

类适配器

使用了继承机制: 适配器同时继承两个对象的接口。
特点

  • Adapter 类直接继承 Service 类,可以在 Adapter 类中对 Service 类的方法进行重定义。
  • 在 Service 类新增抽象方法,Adapter 类也要改动,代码耦合度高。
  • 如果 Service 类有其他子类,Adapter 类中无法调用 Service 子类的方法。
    类适配器URL图

总结

适配器模式结构选择

尽量使用耦合度比较低的对象适配器,少使用类适配器,避免多重继承。但如果构造 Service 对象非常困难时,可以考虑使用类适配器。

适合应用场景

  • 当你希望使用某个类, 但是其接口与其他代码不兼容时, 可以使用适配器类。
  • 如果需要复用这样一些类, 他们处于同一个继承体系, 并且他们又有了额外的一些共同的方法,
    但是这些共同的方法不是所有在这一继承体系中的子类所具有的共性。


    在这里插入图片描述

优点

  • 开闭原则,只要客户端代码通过客户端接口与适配器进行交互,你就能在不修改现有客户端代码的情况下在程序中添加新类型的适配器。
  • 单一职责原则,将接口或数据转换代码从程序主要业务逻辑中分离。

缺点

  • 代码整体复杂度增加,因为你需要新增一系列接口和类。有时直接更改服务类使其与其他代码兼容会更简单。

参考

22种设计模式:refactoringguru.cn/design-patterns
AWTK:github.com/zlgopen/awtk
《设计模式:可复用面向对象软件的基础》

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,884评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,755评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,369评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,799评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,910评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,096评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,159评论 3 411
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,917评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,360评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,673评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,814评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,509评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,156评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,123评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,641评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,728评论 2 351

推荐阅读更多精彩内容