结构型模式--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类图的时候,需要具体问题具体分析,使用相同的设计模式处理不同的业务场景,绘制出的类和类之间的关系也是有些许差别的,不要死读书,读死书,头脑要活泛!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/808163.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

[C++][算法基础]树的重心(树图DFS)

给定一颗树&#xff0c;树中包含 n 个结点&#xff08;编号 1∼n&#xff09;和 n−1 条无向边。 请你找到树的重心&#xff0c;并输出将重心删除后&#xff0c;剩余各个连通块中点数的最大值。 重心定义&#xff1a;重心是指树中的一个结点&#xff0c;如果将这个点删除后&a…

PostgreSQL入门到实战-第十四弹

PostgreSQL入门到实战 PostgreSQL数据过滤(七)官网地址PostgreSQL概述PostgreSQL中BETWEEN 命令理论PostgreSQL中BETWEEN 命令实战更新计划 PostgreSQL数据过滤(七) BETWEEN运算符允许您检查值是否在值的范围内。 官网地址 声明: 由于操作系统, 版本更新等原因, 文章所列内容…

嵌入式网线连接——笔记本电脑设置

一、需求 我们调试很多设备经常需要用到网线去调试&#xff0c;当然主流是USB&#xff0c;和网线。 二、笔记本电脑端设备 有网口的&#xff0c;非常方便&#xff0c;如果没有网口&#xff0c;则需要用到USB转网口 连接指示灯&#xff1a; 绿色&#xff1a;灯亮表示连接正常…

风险评估在应对网络安全威胁中扮演着重要的角色

如今&#xff0c;IT 安全专家面临各种重大威胁&#xff0c;从勒索软件、网络钓鱼&#xff0c;到对基础设施的攻击&#xff0c;再到对知识产权、客户数据的窃取&#xff1b;从不安全的供应链合作伙伴&#xff0c;再到组织内部人员的恶意行为。同时&#xff0c;随着云计算、远程工…

Springboot上传集合,集合超过256直接下标越界

Springboot上传集合&#xff0c;集合超过256直接下标越界 解决方法一 单个controller生效解决方法二 全局controller生效 org.springframework.beans.InvalidPropertyException: Invalid property files[256] of bean class [analysis.vo.wcase.InsertCase]: Invalid list inde…

顺序存储结构的读取、插入与删除

顺序线性表--L已经存在&#xff0c;且1 < index < ListLength(L) 一、获得元素操作--GetElem 含义&#xff1a;将线性表L中的第 index 个位置元素值返回 思路&#xff1a;只要 index 的数值在数组下标值范围内&#xff0c;把数组第 index-1下标的值返回即可 二、插入…

最近公共祖先(LCA)

题目描述 如题,给定一棵有根多叉树,请求出指定两个点直接最近的公共祖先。 输入格式 第一行包含三个正整数 N,M,S,分别表示树的结点个数、询问的个数和树根结点的序号。 接下来 N−1 行每行包含两个正整数x,y,表示 x 结点和 y 结点之间有一条直接连接的边(数据保证可以…

WPF —— 动画缩放变换

ScaleTransform:在二维x-y坐标系统内缩放对象; 在故事板中依赖的属性为RenderTransform.ScaleX或RenderTransform.ScaleY,这要根据你要沿哪个轴进行缩放,X代表x轴,Y代表y轴; key属性当我们使用静态资源访问时候--> <!--TargetType"{x:Type Button} 直接应用…

从零开始做自动驾驶定位(十三)_ 关于建图的讨论

从零开始做自动驾驶定位(十三): 关于建图的讨论 配套代码和测试数据&#xff1a;联系作者获取 附赠自动驾驶学习资料和量产经验&#xff1a;链接 代码在后续可能会有调整&#xff0c;如和文章有出入&#xff0c;以实际代码为准 系列文章进行到这里&#xff0c;建图部分算是基…

详细分析Vuex中的mapGetters

目录 1. 基本知识2. Demo13. Demo2 1. 基本知识 优势和用途 简化代码&#xff1a;用 mapGetters 和 mapState&#xff0c;可以简化组件中对于 Vuex 中状态和 getter 的映射工作&#xff0c;减少了重复的代码书写更易读&#xff1a;组件中直接使用映射的计算属性&#xff0c;使…

Objective-C学习笔记(NSString,NSMutableString,NSArray)4.11

1.框架&#xff1a;系统/第三方写好的类&#xff0c;这些类的集合就是框架。 2.NSString方法&#xff1a;①拼接&#xff1a;stringWithFormat&#xff1a;参数。 ②长度&#xff1a;str.length ③字符串指定下标的字符&#xff1a;characterAtIndex&#xff1a;参数:&#xf…

CSS设置文本

目录 概述&#xff1a; text-aling: text-decoration: text-transform: text-indent: line-height: letter-spacing: word-spacing: text-shadow: vertical-align: white-space: direction: 概述&#xff1a; 在CSS中我们可以设置文本的属性&#xff0c;就像Word文…

20 Games101 - 笔记 - 光场、颜色与感知

**20 ** 光场、颜色与感知 光场 眼睛成像 眼睛成像&#xff1a;我们看到这个三维世界&#xff0c;在眼睛里类似就是一幅二维的图。如果直接看到一幅记录了看到的光线信息的图&#xff0c;也能得到同样效果&#xff08;虚拟现实&#xff09;。 全光函数 全光函数可以描述…

利用国内代理IP突破地域限制访问受限网站实战指南

在某些情况下&#xff0c;由于版权、政策或服务提供商的限制&#xff0c;您可能会遇到无法访问特定网站的问题。这种现象通常被称为“地域限制”。要解决这个问题&#xff0c;一种常用的方法是通过使用国内代理IP来模拟从不同地区进行网络访问&#xff0c;从而绕过这些限制。 以…

Jupyter Notbook如何安装配置并结合内网穿透实现无公网IP远程连接使用

文章目录 推荐1.前言2.Jupyter Notebook的安装2.1 Jupyter Notebook下载安装2.2 Jupyter Notebook的配置2.3 Cpolar下载安装 3.Cpolar端口设置3.1 Cpolar云端设置3.2.Cpolar本地设置 4.公网访问测试5.结语 推荐 前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&am…

Apache—POI详解、小案例展示

简介&#xff1a;Apache POI 是一个处理Miscrosoft Office各种文件格式的开源项目。简单来说就是&#xff0c;我们可以使用POI在Java程序中对Miscrosoft Office各种文件进行读写操作。 目录 1、应用场景 2、案例代码 2.1 创建 Excel 文件 2.2 读取 Excel 文件 1、应用场景 …

java swing个人财务收支管理系统eclipse开发Mysql数据库CS结构java编程

一、源码特点 java swing 个人财务收支管理系统 是一套完善的窗体设计系统&#xff0c;对理解SWING java 编程开发语言有帮助&#xff0c;系统具有完整的源代码和数据库&#xff0c;&#xff0c;系统主要采用C/S模式开发。 应用技术&#xff1a;javamysql 开发工具&#xf…

web APIs总结(1)

1. 根据CSS选择器来获取DOM元素 (重点&#xff09; 获取一个DOM元素我们使用谁&#xff1f;能直接操作修改吗&#xff1f;querySelector() 可以返回值&#xff1a;CSS选择器匹配的第一个元素,一个 HTMLElement对象。如果没有匹配到&#xff0c;则返null 获取多个DOM元素我们使…

有时导数据代码比写SQL要快很多

一、背景 接到一个数据需求&#xff0c;是从我们的Mongo中导出的&#xff0c;但要取的值得到很深的层级&#xff0c;尝试写了半天Mongo的查询查不出来&#xff0c;问了半天大模型给的也不对&#xff0c;于是考虑写代码的方式 二、数据格式 // 1 {"_id": ObjectId(&…

FastAPI+Sqlalchemy执行【Mysql】原生sql

一. 前言 当有的复杂sql 用orm不好写出来的时候&#xff0c;此时想要用原生sql查询 原生sql查询&#xff0c;查出的结果是对象原生sql查询&#xff0c;查询结果列表套元组 当你使用 SQLAlchemy 并希望通过创建的 session 对象来执行原生 SQL 语句时&#xff0c;可以使用 ses…