【深入理解设计模式】适配器设计模式

适配器设计模式
适配器设计模式是一种结构型设计模式,用于将一个类的接口转换成客户端所期望的另一个接口,从而使得原本由于接口不兼容而不能一起工作的类能够一起工作。适配器模式通常用于以下场景:
-
现有接口与需求不匹配:当需要使用的类的接口与当前系统的接口不匹配时,可以创建一个适配器来进行转换。
-
类的功能需要增强:有时候,为了增强现有类的功能而不修改原有代码,可以使用适配器模式。
概述
如果去欧洲国家去旅游的话,他们的插座如下图最左边,是欧洲标准。而我们使用的插头如下图最右边的。因此我们的笔记本电脑,手机在当地不能直接充电。所以就需要一个插座转换器,转换器第1面插入当地的插座,第2面供我们充电,这样使得我们的插头在当地能使用。生活中这样的例子很多,手机充电器(将220v转换为5v的电压),读卡器等,其实就是使用到了适配器模式。

定义:
将一个类的接口转换成客户端希望的另外一个接口,使得原本由于接口不兼容而不能一起工作的那些类能一起工作。
适配器模式分为类适配器模式和对象适配器模式,前者类之间的耦合度比后者高,且要求程序员了解现有组件库中的相关组件的内部结构,所以应用相对较少些。
结构
适配器模式(Adapter)包含以下主要角色:
- 目标(Target)接口:当前系统业务所期待的接口,它可以是抽象类或接口。
- 被适配者(Adaptee):需要被适配的类。它定义了适配器所需的原始接口。
- 适配器(Adapter)类:它是一个转换器,通过继承或引用适配者的对象,把适配者接口转换成目标接口,让客户按目标接口的格式访问适配者。
适配器模式的实现可以分为两种方式:
-
类适配器模式:通过继承被适配类和实现目标接口来实现适配器。这种方式需要多重继承或接口实现,不过在一些编程语言中并不支持多重继承,因此并不常用。
-
对象适配器模式:通过在适配器类中组合或聚合被适配类的实例来实现适配器。这种方式更加灵活,因为它可以适配多个类而不仅限于单一类。
适配器设计模式能够很好地解决不同接口之间的兼容性问题,使得原本不兼容的类能够协同工作,提高了代码的复用性和灵活性。
类适配器模式
实现方式:定义一个适配器类来实现当前系统的业务接口,同时又继承现有组件库中已经存在的组件。
【例】读卡器
现有一台电脑只能读取SD卡,而要读取TF卡中的内容的话就需要使用到适配器模式。创建一个读卡器,将TF卡中的内容读取出来。
代码如下:
/*** @author OldGj 2024/02/23* @version v1.0* @apiNote 电脑类*/
public class Computer {// 向SD卡中写数据(目标接口只能是SDCard)public void writeSD(SDCard sdCard, String msg) {sdCard.writeMsg(msg);}// 从SD卡中读取数据(目标接口只能是SDCard)public String readSD(SDCard sdCard) {return sdCard.readMsg();}
}
/*** @author OldGj 2024/02/23* @version v1.0* @apiNote 目标接口 - SD卡接口*/
public interface SDCard {// 向SD卡中写数据void writeMsg(String msg);// 从SD卡中读数据String readMsg();}
/*** @author OldGj 2024/02/23* @version v1.0* @apiNote 目标接口实现类 - SDCard实现类*/
public class SDCardImpl implements SDCard{@Overridepublic void writeMsg(String msg) {System.out.println("write msg to sd card"+msg);}@Overridepublic String readMsg() {return "read msg from sd card";}
}
/*** @author OldGj 2024/02/23* @version v1.0* @apiNote 适配者 - TFCard*/
public interface TFCard {// 向TF卡中写数据void writeMsg(String msg);// 从TF卡中读数据String readMsg();}
/*** @author OldGj 2024/02/23* @version v1.0* @apiNote 具体适配者 - TFCard具体实现类*/
public class TFCardImpl implements TFCard{@Overridepublic void writeMsg(String msg) {System.out.println("write msg to tf card"+msg);}@Overridepublic String readMsg() {return "read msg from tf card";}
}
/*** @author OldGj 2024/02/23* @version v1.0* @apiNote 适配器类 继承被适配者类 实现目标接口*/
public class SDAdapterTF extends TFCardImpl implements SDCard{@Overridepublic void writeMsg(String msg) {super.writeMsg(msg);}@Overridepublic String readMsg() {return super.readMsg();}
}
/*** @author OldGj 2024/02/23* @version v1.0* @apiNote 客户端 - 测试类*/
public class Client {public static void main(String[] args) {Computer computer = new Computer();SDCard sdCard = new SDCardImpl();String s = computer.readSD(sdCard);System.out.println(s);System.out.println("=============");// 通过适配器,将TF卡转换为实现SD接口的适配器SDAdapterTF adapterTF = new SDAdapterTF();// 读取适配器中的数据String s1 = computer.readSD(adapterTF);System.out.println(s1);}
}
类适配器模式违背了合成复用原则。类适配器是客户类有一个接口规范的情况下可用,反之不可用。
对象适配器模式
实现方式:对象适配器模式可釆用将现有组件库中已经实现的组件引入适配器类中,该类同时实现当前系统的业务接口。
【例】读卡器
我们使用对象适配器模式将读卡器的案例进行改写。
类适配器模式的代码,我们只需要修改适配器类(SDAdapterTF)和测试类。
/*** @author OldGj 2024/02/23* @version v1.0* @apiNote 适配器类 继承适配者类 实现目标接口*/
public class SDAdapterTF implements SDCard {private TFCard tfCard;public SDAdapterTF(TFCard tfCard) {this.tfCard = tfCard;}@Overridepublic void writeMsg(String msg) {tfCard.writeMsg(msg);}@Overridepublic String readMsg() {return tfCard.readMsg();}
}
/*** @author OldGj 2024/02/23* @version v1.0* @apiNote 客户端 - 测试类*/
public class Client {public static void main(String[] args) {Computer computer = new Computer();SDCard sdCard = new SDCardImpl();String s = computer.readSD(sdCard);System.out.println(s);System.out.println("=============");// 将TF卡传入适配器中SDAdapterTF sdAdapterTF = new SDAdapterTF(new TFCardImpl());String s1 = computer.readSD(sdAdapterTF);System.out.println(s1);}
}
注意:还有一个适配器模式是接口适配器模式。当不希望实现一个接口中所有的方法时,可以创建一个抽象类Adapter ,实现所有方法。而此时我们只需要继承该抽象类即可。
应用场景
- 以前开发的系统存在满足新系统功能需求的类,但其接口同新系统的接口不一致。
- 使用第三方提供的组件,但组件接口定义和自己要求的接口定义不同。
JDK中的适配器模式:
当涉及字符流(Reader)和字节流(InputStream)之间的适配时,通常会使用适配器模式。这种适配器模式的目的是让字符流和字节流能够协同工作,尽管它们的接口不同。
-
Reader(字符流):
Reader是 Java 中用于读取字符流的抽象基类。它定义了读取字符数据的一系列方法,如read()、close()等。字符流是以字符为单位进行读取和写入的,对文本数据的处理更为方便。 -
InputStream(字节流):
InputStream是 Java 中用于读取字节流的抽象基类。它定义了读取字节数据的一系列方法,如read()、close()等。字节流是以字节为单位进行读取和写入的,适用于处理二进制数据。 -
InputStreamReader(适配器):
InputStreamReader是 Java 中用于将字节流转换为字符流的适配器类。它实现了Reader接口,并包装了一个InputStream对象。InputStreamReader通过在字节流和字符流之间进行转换,使得字符流能够读取字节流中的数据。它的作用就是将字节流适配成字符流,使得原本不兼容的字符流和字节流能够一起工作。
在使用 InputStreamReader 时,它会接受一个 InputStream 对象作为参数,并将该对象转换为字符流,因此它充当了字符流和字节流之间的适配器。这样一来,当我们需要使用字符流操作时,可以直接使用 Reader 接口及其实现类,而不必直接操作字节流。InputStreamReader 负责将底层的字节流适配成字符流,从而实现了字符流和字节流之间的适配。
总而言之,Reader 和 InputStream 之间的适配器模式的典型应用就是通过 InputStreamReader 将字节流适配成字符流,使得字符流和字节流能够协同工作,这是适配器模式在 Java IO 中的一个典型应用。
相关文章:
【深入理解设计模式】适配器设计模式
适配器设计模式 适配器设计模式是一种结构型设计模式,用于将一个类的接口转换成客户端所期望的另一个接口,从而使得原本由于接口不兼容而不能一起工作的类能够一起工作。适配器模式通常用于以下场景: 现有接口与需求不匹配:当需要…...
ASP.NET-实现图形验证码
ASP.NET 实现图形验证码能够增强网站安全性,防止机器人攻击。通过生成随机验证码并将其绘制成图像,用户在输入验证码时增加了人机交互的难度。本文介绍了如何使用 C# 和 ASP.NET 创建一个简单而有效的图形验证码系统,包括生成随机验证码、绘制…...
解决Maven爆红以及解决 Idea 卡在 Resolving问题
关于 Idea 卡在 Resolving(前提是Maven的setting.xml中配置好了阿里云和仓库) 参考文章https://blog.csdn.net/jiangyu1013/article/details/95042611 解决Maven爆红参考文章https://devpress.csdn.net/beijing/656d993b76f0791b6eca7bb0.html?dp_toke…...
MySQL集群 双主架构(配置命令)
CSDN 成就一亿技术人! 今天刚开学第一天给大家分享一期:MySQL集群双主的配置需求和命令 CSDN 成就一亿技术人! 神秘泣男子主页:作者首页 <———— MySQL专栏 :MySQL数据库专栏<———— MySQL双主是一…...
网络安全之安全事件监测
随着人们对技术和智能互联网设备依赖程度的提高,网络安全的重要性也在不断提升。因此,我们需要不断加强网络安全意识和措施,确保网络环境的安全和稳定。 网络安全的重要性包含以下几点: 1、保护数据安全:数据是组织和…...
【BUG 记录】MyBatis-Plus 处理枚举字段和 JSON 字段
【BUG 记录】MyBatis-Plus 处理枚举字段和JSON字段 一、枚举字段(mysql环境已测、postgresql环境已测)1.1 场景1.2 定义枚举常量1.3 配置枚举处理器1.4 测试 二、JSON字段(mysql环境已测)2.1 导包2.2 使用对象接受2.3 测试 三、JS…...
Web性能优化-详细讲解与实用方法-MDN文档学习笔记
Web性能优化 查看更多学习笔记:GitHub:LoveEmiliaForever MDN中文官网 性能优良的网站能够提高访问者留存和用户满意度,减少客户端和服务器之间传输的数据量可降低各方的成本 不同的业务目标和用户需求需要不同的性能度量,要提高…...
组态王连接施耐德M580PLC
组态王连接施耐德M580 网络架构 网线连接PLC和装组态王软件的PC组态设置帮助 可先查看帮助:菜单栏点击【帮助】->【驱动帮助】,在弹出窗口中PLC系列选择莫迪康PLC的“modbusRtu\ASSCII\TCP”查看组态配置流程: 相关说明: 1、…...
pop链构造 [NISACTF 2022]babyserialize
打开题目 题目源代码如下 <?php include "waf.php"; class NISA{public $fun"show_me_flag";public $txw4ever;public function __wakeup(){if($this->fun"show_me_flag"){hint();}}function __call($from,$val){$this->fun$val[0];…...
【VIP专属】Python应用案例——基于Keras, OpenCV和MobileNet口罩佩戴识别
目录 1、导入所需库 2、加载人脸口罩检测数据集 3、对标签进行独热编码...
Doris——荔枝微课统一实时数仓建设实践
目录 一、业务介绍 二、早期架构及痛点 2.1 早期架构 2.2 架构痛点 三、技术选型 四、新的架构及方案 五、搭建经验 5.1 数据建模 5.2 数据开发 5.3 库表设计 5.4 数据管理 5.4.1 监控告警 5.4.2 数据备份与恢复 六、收益总结 七、未来规划 原文大佬这篇Doris腾…...
Stable Diffusion 绘画入门教程(webui)-ControlNet(Inpaint)
上篇文章介绍了语义分割Tile/Blur,这篇文章介绍下Inpaint(重绘) Inpaint类似于图生图的局部重绘,但是Inpain效果要更好一点,和原图融合会更加融洽,下面是案例,可以看下效果(左侧原图…...
LeetCode146: LRU缓存
题目描述 请你设计并实现一个满足 LRU (最近最少使用) 缓存 约束的数据结构。 实现 LRUCache 类: LRUCache(int capacity) 以 正整数 作为容量 capacity 初始化 LRU 缓存 int get(int key) 如果关键字 key 存在于缓存中,则返回关键字的值,否则…...
【ArcGIS】基于DEM/LUCC等数据统计得到各集水区流域特征
基于DEM/LUCC等数据统计得到各集水区流域特征 提取不同集水区各类土地利用类型比例步骤1:划分集水区为独立面单元步骤2:批量掩膜提取得到各集水区土地利用类型比例步骤3:导入各集水区LUCC数据并统计得到各类型占比 提取坡度特征流域面坡度河道…...
vue3中安装并使用CSS预处理器Sass的方法介绍
文章目录 Sass是什么?为什么使用Sass?安装sass1、安装sass2、编写全局css变量/全局mixin3、vite引入并使用4、按需引入并使用 sass语法1、变量创建一个变量使用变量变量作用域 2、数学计算两个Sass有关于数学计算的“陷阱” 3、嵌套4、Imports sass中文官网 Sass是…...
过滤器(Filter)
过滤器(Filter) 1. 基本概念 过滤器(Filter)是拦截 Request 请求的对象:在用户的请求访问资源前处理 ServletRequest 和 ServletResponse 。 Filter 相关的接口有:Filter、FilterConfig、FilterChain 。…...
AMRT3D数字孪生引擎详解
AMRT 3D数字孪生引擎介绍 AMRT3D引擎是一款融合了眸瑞科技的AMRT格式与轻量化处理技术为基础,以降本增效为目标,支持多端发布的一站式纯国产自研的CS架构项目开发引擎。 引擎包括场景搭建、UI拼搭、零代码交互事件、光影特效组件、GIS/BIM组件、实时数据…...
Sqlite数据库详解
1.关于Sqlite SQLite 是一个进程内库,它实现了一个独立的、无服务器的、零配置的事务性 SQL 数据库引擎。 SQLite的代码属于公共领域,因此对 用于任何目的,商业或私人目的。 SQLite是世界上部署最广泛的数据库 应用程序比我们能做的要多 计数…...
基于YOLOv8深度学习+Pyqt5的电动车头盔佩戴检测系统
wx供重浩:创享日记 对话框发送:225头盔 获取完整源码源文件已标注的数据集(1463张)源码各文件说明配置跑通说明文档 若需要一对一远程操作在你电脑跑通,有偿59yuan 效果展示 基于YOLOv8深度学习PyQT5的电动车头盔佩戴检…...
【数据结构】B树,B+树,B*树
文章目录 一、B树1.B树的定义2.B树的插入3.B树的中序遍历 二、B树和B*树1.B树的定义2.B树的插入3.B*树的定义4.B树系列总结 三、B树与B树的应用 一、B树 1.B树的定义 1. 在内存中搜索效率高的数据结构有AVL树,红黑树,哈希表等,但这是在内存…...
立创EDA画STM32板子,这些“隐藏”设置能让你的PCB一次打样成功
立创EDA画STM32板子的7个高阶设置技巧 第一次用立创EDA画完STM32板子的那种成就感,往往会被打样回来后发现的问题冲淡——电源线发热、信号干扰、过孔断裂...这些问题大多源于一些容易被忽略的参数设置。作为用过上百次立创EDA的老手,我总结出这些实战经…...
3步轻松搞定PDF智能书签:告别无序阅读,拥抱高效导航
3步轻松搞定PDF智能书签:告别无序阅读,拥抱高效导航 【免费下载链接】pdfdir PDF导航(大纲/目录)添加工具 项目地址: https://gitcode.com/gh_mirrors/pd/pdfdir 还在为没有书签的PDF电子书而烦恼吗?每次翻阅都…...
绩效谈判技巧:如何让老板为你的技术价值买单
在软件研发的生态中,测试工程师的角色常常处于一种微妙的“价值隐形”状态。开发构建功能,运维保障稳定,而测试——在许多管理者眼中——似乎只是流程中一个“找问题”的环节,其价值容易被量化为发现的缺陷数量,却难以…...
Bilibili评论爬虫:零基础获取B站完整评论数据的终极指南
Bilibili评论爬虫:零基础获取B站完整评论数据的终极指南 【免费下载链接】BilibiliCommentScraper B站视频评论爬虫 Bilibili完整爬取评论数据,包括一级评论、二级评论、昵称、用户ID、发布时间、点赞数 项目地址: https://gitcode.com/gh_mirrors/bi/…...
完全掌握Windows Cleaner:高效使用开源系统优化工具深度解析
完全掌握Windows Cleaner:高效使用开源系统优化工具深度解析 【免费下载链接】WindowsCleaner Windows Cleaner——专治C盘爆红及各种不服! 项目地址: https://gitcode.com/gh_mirrors/wi/WindowsCleaner Windows Cleaner是一款专为Windows系统设…...
抖音评论采集终极指南:3步零代码实现自动化数据抓取
抖音评论采集终极指南:3步零代码实现自动化数据抓取 【免费下载链接】TikTokCommentScraper 项目地址: https://gitcode.com/gh_mirrors/ti/TikTokCommentScraper 还在为手动复制抖音评论而烦恼吗?想要批量获取视频评论数据却苦于没有技术背景&a…...
告别F8乱按!用OllyDbg调试破解一个简单注册框的保姆级实录
从零开始用OllyDbg破解简单注册框:新手逆向实战指南 逆向工程的世界总是充满神秘感,而OllyDbg(简称OD)就像一把打开这扇大门的金钥匙。记得我第一次接触逆向时,面对满屏的十六进制代码和汇编指令,那种既兴奋…...
别再纠结选哪种激光器了!一张图看懂CO2、光纤、半导体、YAG、碟片激光器怎么选(附应用场景对比)
工业激光器选型实战指南:5大类型核心差异与应用场景解析 当车间主任老张第三次修改采购清单时,他的不锈钢样品正静静躺在三种激光切割机的测试台上。这个场景每天都在全球数以万计的工厂里上演——面对CO2激光器切割亚克力时的完美断面,光纤激…...
别再死记硬背UART帧格式了!用Verilog手撕一个收发器,彻底搞懂起始位、波特率与采样
用Verilog手撕UART收发器:从状态机到上板调试的实战指南 在数字电路设计中,UART协议就像一位沉默的邮差——它不需要时钟线同步,仅凭两根信号线就能完成设备间的对话。但正是这种简洁性,让许多初学者在理解其底层机制时陷入困惑&a…...
猫抓浏览器扩展:轻松获取M3U8流媒体和在线视频的终极指南
猫抓浏览器扩展:轻松获取M3U8流媒体和在线视频的终极指南 【免费下载链接】cat-catch 猫抓 浏览器资源嗅探扩展 / cat-catch Browser Resource Sniffing Extension 项目地址: https://gitcode.com/GitHub_Trending/ca/cat-catch 你是否经常遇到想下载在线视频…...
