结构型模式--1.适配器模式【托尼托尼·乔巴】

2024-04-10 06:12

本文主要是介绍结构型模式--1.适配器模式【托尼托尼·乔巴】,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1. 翻译家

在海贼王中,托尼托尼·乔巴(Tony Tony Chopper)草帽海贼团的船医,它本来是一头驯鹿,但是误食了动物系·人人果实之后可以变成人的形态。

在这里插入图片描述

乔巴吃了恶魔果实之后的战斗力暂且抛开不谈,说说它掌握的第二技能:语言,此时的他既能听懂人的语言,又能听懂动物语言,妥妥的语言学家。

人和动物本来无法直接交流,但是有了乔巴的存在,就相当于了有了一条纽带,一座桥梁,使得二者之间能够顺畅的沟通。在这里边,乔巴充当的就是一个适配器,将一个类的接口转换成用户希望的另一个接口,使不兼容的对象能够相互配合并一起工作,这种模式就叫适配器模式。说白了,适配器模式就相当于找了一个翻译。

需要适配器的例子或者场景很多,随便列举几个:

  1. STL标准模板库有六大组件,其中之一的就是适配器。
    六大组件分别是:容器、算法、迭代器、仿函数、适配器、空间适配器。
    适配器又可以分为:容器适配器、函数适配器、迭代器适配器
  2. 台湾省的电压是110V,大陆是220V,如果他们把大陆的电器带回台湾就需要适配器进行电压的转换。
  3. 香港的插座插孔是欧式的,从大陆去香港旅游,就需要带转换头(适配器)
  4. 儿媳妇儿和婆婆打架,就需要儿子从中调解,此时儿子就适配器。
  5. 手机、平板、电脑等需要的电压并不是220V,也需要适配器进行转换。

2. 斜杠型人才

所谓的斜杠型人才就是多才多艺,适配器也一样,如果它能给多个不相干的对象进行相互之间的适配,这个适配器就是斜杠适配器。

还是拿乔巴举例子,他既能把人的语言翻译给动物,又能把动物的语言翻译给人,那么此时的乔巴就是一个双向适配器。

2.1 国宝的血泪史

覆巢之下无完卵,清朝末年,作为曾经蚩尤的坐骑而今呆萌可爱的国宝大熊猫惨遭西方国家围猎和杀戮,其中以美利坚尤甚。

1869年春天,法国传教士阿尔芒·戴维在四川宝兴寻找珍稀物种的时候发现了大熊猫,国宝的厄运就此开始。

这是美国总统罗斯福两个儿子制作的大熊猫标本:
在这里插入图片描述

那个时候的旧中国对熊猫还没有一个确切的认识,并没有采取任何措施,于是外国人对熊猫的猎杀活动更变本加厉起来。据统计,在1936年到1946年之间,超过16只活体大熊猫从中国运出,而熊猫标本更是多达70余具!
在这里插入图片描述

一切终成过去,作为一个程序猿我也不能改变什么,但是我决定要写个程序让这群混蛋给被他们杀死的国宝道歉!

2.2 抽丝剥茧

对于杀害大熊猫的这群西方的混蛋,不论他们怎么忏悔大熊猫肯定也是听不懂的,所以需要适配器来完成这二者之间的交流。但是这帮人来自不同的国家,它们都有自己的语言,所以这个适配器翻译的不是一种人类语言而是多种,因此我们要做如下的处理:

  1. 忏悔的人说的是不同的语言,所以需要一个抽象类。
  2. 适配器需要翻译不同国家的语言,所以适配器也需要一个抽象类。
  3. 西方人需要给大熊猫道歉,因此需要给大熊猫定义一个类。

西方罪人

先把西方这群烂人对应的类定义出来:

class Foreigner
{
public:virtual string confession() = 0;void setResult(string msg){cout << "Panda Say: " << msg << endl;}virtual ~Foreigner() {}
};// 美国人
class American : public Foreigner
{
public:string confession() override{return string("我是畜生, 我有罪!!!");}
};// 法国人
class French : public Foreigner
{
public:string confession(){return string("我是强盗, 我该死!!!");}
};

不同国家的西方罪人需要使用不同的语言向大熊猫忏悔,所以美国人法国人作为子类需要重写从父类继承的用于忏悔的虚函数confession()

当乔巴这个适配器翻译了熊猫的语言之后,需要通过void setResult(string msg)函数将信息传递给西方罪人对象。

大熊猫

再把国宝对应的类定义出来

// 大熊猫
class Panda
{
public:void recvMessage(string msg){cout << msg << endl;}string sendMessage(){return string("强盗、凶手、罪人是不可能被宽恕和原谅的!");}
};

大熊猫类有两个方法:

  • recvMessage(string msg):接收忏悔信息。
  • string sendMessage():告诉西方人是否原谅他们。

乔巴登场

同时能听懂人类和动物语言非乔巴莫属,由于要翻译两种不同的人类语言,所以需要一个抽象的乔巴适配器类,在其子类中完成英语 <==> 熊猫语、法语 <==> 熊猫语之间的翻译。

// 抽象乔巴适配器类
class AbstractChopper
{
public:AbstractChopper(Foreigner* foreigner) : m_foreigner(foreigner) {}virtual void translateToPanda() = 0;virtual void translateToHuman() = 0;virtual ~AbstractChopper() {}
protected:Panda m_panda;Foreigner* m_foreigner = nullptr;
};// 英语乔巴适配器
class EnglishChopper : public AbstractChopper
{
public:// 继承构造函数using AbstractChopper::AbstractChopper;void translateToPanda() override{string msg = m_foreigner->confession();// 翻译并将信息传递给熊猫对象m_panda.recvMessage("美国人说: " + msg);}void translateToHuman() override{// 接收熊猫的信息string msg = m_panda.sendMessage();// 翻译并将熊猫的话转发给美国人m_foreigner->setResult("美国佬, " + msg);}
};// 法语乔巴适配器
class FrenchChopper : public AbstractChopper
{
public:using AbstractChopper::AbstractChopper;void translateToPanda() override{string msg = m_foreigner->confession();// 翻译并将信息传递给熊猫对象m_panda.recvMessage("法国人说: " + msg);}void translateToHuman() override{// 接收熊猫的信息string msg = m_panda.sendMessage();// 翻译并将熊猫的话转发给法国人m_foreigner->setResult("法国佬, " + msg);}
};

在上面的适配器类中,同时访问了Foreigner 类Panda 类,这样适配器类就可以拿到这两个类对象中的数据进行转译,最后再将其分别发送给对方,这样这两个不相干的没有交集的类对象之间就可以正常的沟通了。

不可原谅

最后编写程序进行测试,这部分程序其实是通过客户端的操作并被执行的,此处就将其直接写到main()函数中了:

int main()
{Foreigner* human = new American;EnglishChopper* american = new EnglishChopper(human);american->translateToPanda();american->translateToHuman();delete human;delete american;human = new French;FrenchChopper* french = new FrenchChopper(human);french->translateToPanda();french->translateToHuman();delete human;delete french;return 0;
}

程序输出的结果:

美国人说: 我是畜生, 我有罪!!!
Panda Say: 美国佬, 强盗、凶手、罪人是不可能被宽恕和原谅的!
============================
法国人说: 我是强盗, 我该死!!!
Panda Say: 法国佬, 强盗、凶手、罪人是不可能被宽恕和原谅的!

3. 结构图

最后根据上面的代码,把对应的UML类图画一下(再次强调,UML类图是在在写程序之前画的,用来梳理程序的设计思路。学会了设计模式之后,就需要在写程序之前画类图了。)

在这里插入图片描述

在这个UML类图中,将抽象的乔巴类(抽象适配器类)熊猫类设置为了关联关系,除了使用这种方式我们还可以让抽象的适配器类继承熊猫类,这样在适配器类中就可以直接使用熊猫类中定义的方法了,如下图:

在这里插入图片描述

上图对应的代码如下:

class Foreigner
{
public:virtual string confession() = 0;void setResult(string msg){cout << "Panda Say: " << msg << endl;}virtual ~Foreigner() {}
};// 美国人
class American : public Foreigner
{
public:string confession() override{return string("我是畜生, 我有罪!!!");}
};// 法国人
class French : public Foreigner
{
public:string confession(){return string("我是强盗, 我该死!!!");}
};// 大熊猫
class Panda
{
public:void recvMessage(string msg){cout << msg << endl;}string sendMessage(){return string("强盗、凶手、罪人是不可能被宽恕和原谅的!");}
};// 抽象适配器类
class AbstractChopper : public Panda
{
public:AbstractChopper(Foreigner* foreigner) : m_foreigner(foreigner) {}virtual void translateToPanda() = 0;virtual void translateToHuman() = 0;virtual ~AbstractChopper() {}
protected:Foreigner* m_foreigner = nullptr;
};class EnglishChopper : public AbstractChopper
{
public:using AbstractChopper::AbstractChopper;void translateToPanda() override{string msg = m_foreigner->confession();// 翻译并将信息传递给熊猫对象recvMessage("美国人说: " + msg);}void translateToHuman() override{// 接收熊猫的信息string msg = sendMessage();// 翻译并将熊猫的话转发给美国人m_foreigner->setResult("美国佬, " + msg);}
};class FrenchChopper : public AbstractChopper
{
public:using AbstractChopper::AbstractChopper;void translateToPanda() override{string msg = m_foreigner->confession();// 翻译并将信息传递给熊猫对象recvMessage("法国人说: " + msg);}void translateToHuman() override{// 接收熊猫的信息string msg = sendMessage();// 翻译并将熊猫的话转发给法国人m_foreigner->setResult("法国佬, " + msg);}
};int main()
{Foreigner* human = new American;EnglishChopper* american = new EnglishChopper(human);american->translateToPanda();american->translateToHuman();delete human;delete american;cout << "============================" << endl;human = new French;FrenchChopper* french = new FrenchChopper(human);french->translateToPanda();french->translateToHuman();delete human;delete french;return 0;
}

上面的代码和第一个版本的代码其实是没有太大区别,如果仔细观察会发现,在适配器类中使用熊猫类中的方法的时候就无需通过熊猫类的对象来调用了,因为适配器类变成了熊猫类的子类,把这些方法继承下来了。

使用这样的模型结构,有一点需要注意:如果熊猫类有子类,那么还是建议将熊猫类和适配器类设置为关联关系

其实关于适配器模式还有另外的一种实现方式:就是让适配器类继承它要为之提供服务器的类,也就是这个例子中的外国人类和熊猫类(如果外国人来没有子类可以使用这种方式),这种解决方案要求使用的面向对象的语言支持多继承,对于这一点C++是满足要求的,但是很多其它面向对象的语言不支持多继承。

再次强调,在使用适配器类为相关的类提供适配服务的时候,如果这个类没有子类就可以让适配器类继承这个类,如果这个类有子类,此时使用继承就不太合适了,建议将适配器类和要被适配的类设置为关联关系。

在画UML类图的时候,需要具体问题具体分析,使用相同的设计模式处理不同的业务场景,绘制出的类和类之间的关系也是有些许差别的,不要死读书,读死书,头脑要活泛!

这篇关于结构型模式--1.适配器模式【托尼托尼·乔巴】的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/890255

相关文章

在JS中的设计模式的单例模式、策略模式、代理模式、原型模式浅讲

1. 单例模式(Singleton Pattern) 确保一个类只有一个实例,并提供一个全局访问点。 示例代码: class Singleton {constructor() {if (Singleton.instance) {return Singleton.instance;}Singleton.instance = this;this.data = [];}addData(value)

模版方法模式template method

学习笔记,原文链接 https://refactoringguru.cn/design-patterns/template-method 超类中定义了一个算法的框架, 允许子类在不修改结构的情况下重写算法的特定步骤。 上层接口有默认实现的方法和子类需要自己实现的方法

【iOS】MVC模式

MVC模式 MVC模式MVC模式demo MVC模式 MVC模式全称为model(模型)view(视图)controller(控制器),他分为三个不同的层分别负责不同的职责。 View:该层用于存放视图,该层中我们可以对页面及控件进行布局。Model:模型一般都拥有很好的可复用性,在该层中,我们可以统一管理一些数据。Controlller:该层充当一个CPU的功能,即该应用程序

迭代器模式iterator

学习笔记,原文链接 https://refactoringguru.cn/design-patterns/iterator 不暴露集合底层表现形式 (列表、 栈和树等) 的情况下遍历集合中所有的元素

《x86汇编语言:从实模式到保护模式》视频来了

《x86汇编语言:从实模式到保护模式》视频来了 很多朋友留言,说我的专栏《x86汇编语言:从实模式到保护模式》写得很详细,还有的朋友希望我能写得更细,最好是覆盖全书的所有章节。 毕竟我不是作者,只有作者的解读才是最权威的。 当初我学习这本书的时候,只能靠自己摸索,网上搜不到什么好资源。 如果你正在学这本书或者汇编语言,那你有福气了。 本书作者李忠老师,以此书为蓝本,录制了全套视频。 试

利用命令模式构建高效的手游后端架构

在现代手游开发中,后端架构的设计对于支持高并发、快速迭代和复杂游戏逻辑至关重要。命令模式作为一种行为设计模式,可以有效地解耦请求的发起者与接收者,提升系统的可维护性和扩展性。本文将深入探讨如何利用命令模式构建一个强大且灵活的手游后端架构。 1. 命令模式的概念与优势 命令模式通过将请求封装为对象,使得请求的发起者和接收者之间的耦合度降低。这种模式的主要优势包括: 解耦请求发起者与处理者

springboot实战学习(1)(开发模式与环境)

目录 一、实战学习的引言 (1)前后端的大致学习模块 (2)后端 (3)前端 二、开发模式 一、实战学习的引言 (1)前后端的大致学习模块 (2)后端 Validation:做参数校验Mybatis:做数据库的操作Redis:做缓存Junit:单元测试项目部署:springboot项目部署相关的知识 (3)前端 Vite:Vue项目的脚手架Router:路由Pina:状态管理Eleme

状态模式state

学习笔记,原文链接 https://refactoringguru.cn/design-patterns/state 在一个对象的内部状态变化时改变其行为, 使其看上去就像改变了自身所属的类一样。 在状态模式中,player.getState()获取的是player的当前状态,通常是一个实现了状态接口的对象。 onPlay()是状态模式中定义的一个方法,不同状态下(例如“正在播放”、“暂停

软件架构模式:5 分钟阅读

原文: https://orkhanscience.medium.com/software-architecture-patterns-5-mins-read-e9e3c8eb47d2 软件架构模式:5 分钟阅读 当有人潜入软件工程世界时,有一天他需要学习软件架构模式的基础知识。当我刚接触编码时,我不知道从哪里获得简要介绍现有架构模式的资源,这样它就不会太详细和混乱,而是非常抽象和易

C++ STL 适配器

系列文章目录 模板特例化,偏特化,左右值引用 https://blog.csdn.net/surfaceyan/article/details/126794013 C++ STL 关联容器 https://blog.csdn.net/surfaceyan/article/details/127414434 C++ STL 序列式容器(二) https://blog.csdn.net/surfac