设计模式-结构型-适配器模式
在软件开发中,随着系统的不断扩展和模块的不断增加,往往会遇到不同模块之间接口不兼容的情况。此时,如果我们能通过某种方式将一个接口转化为另一个接口,那么开发工作将变得更加灵活和高效。适配器模式(Adapter Pattern)正是为了解决这个问题而设计的,它能够让两个不兼容的接口能够协同工作。
本文将深入解析适配器模式的概念、应用场景、实现方法,并通过代码示例帮助大家理解如何在实际开发中使用适配器模式。
1. 适配器模式简介
适配器模式属于结构型模式,它通过将一个类的接口转换为客户端所期待的另一个接口,使得原本接口不兼容的类可以一起工作。
定义:
适配器模式(Adapter Pattern)是通过“适配器类”对两个接口进行转换,从而使得不兼容的接口能够正常交互和工作。简单来说,它充当了一个桥梁,使得接口不匹配的类能够协同工作。
2. 适配器模式的结构
适配器模式主要包含以下几个部分:
- 目标接口(Target):客户端所需要的接口,它是适配器模式的核心。
- 源接口(Adaptee):需要被适配的现有接口。这个接口与目标接口不兼容,需要通过适配器进行转换。
- 适配器(Adapter):适配器实现了目标接口,调用源接口的相关方法,以达到兼容的效果。
UML 图示
图像来源
- Client:客户端,调用目标接口的代码。
- Target:目标接口,客户端期待的接口。
- Adapter:适配器类,适配源接口(Adaptee)到目标接口(Target)。
- Adaptee:源接口,原有的不兼容接口。
3. 适配器模式的类型
适配器模式可以分为两种常见的类型,具体选择哪种取决于应用场景:
3.1 类适配器模式(Class Adapter)
类适配器通过继承源类(Adaptee)来实现目标接口(Target)。在类适配器中,适配器通过继承的方式来实现接口的适配。
3.2 对象适配器模式(Object Adapter)
对象适配器通过组合(而非继承)来实现目标接口,适配器将源对象作为成员变量,并通过调用源对象的方法来完成适配。这种方式更加灵活,因为它不依赖于类的继承关系。
4. 适配器模式的应用场景
适配器模式通常用于以下几种情况:
-
需要复用现有类的功能,但接口不兼容时: 如果已有的类或者模块接口与当前系统中的接口不兼容,可以使用适配器模式来进行适配。
-
系统中多个类需要转换成统一接口时: 如果系统中有多个类,客户端希望统一调用它们的接口,而这些类的接口不同,可以通过适配器模式进行统一适配。
-
第三方库的接口与系统不兼容时: 在集成第三方库时,库的接口可能与现有系统的接口不匹配,适配器模式可以帮助进行接口转换。
5. 适配器模式的优缺点
优点:
- 解耦:通过适配器模式,客户端和原有接口之间的耦合度降低,客户端不再依赖于源接口的实现细节。
- 提高兼容性:适配器模式使得两个不兼容的接口能够协同工作,扩展性好。
- 代码复用性强:可以将已有类的功能复用到新的接口上,避免重复代码。
缺点:
- 增加系统复杂性:由于引入了适配器类,可能会增加代码的复杂度,导致系统结构变得更加复杂。
- 性能开销:适配器模式需要额外的对象包装和转换,可能会导致一定的性能开销。
6. 适配器模式的实现
6.1 示例:类适配器模式
假设我们有一个需求,需要将一个旧的类(OldSystem
)的接口适配到新的系统接口(TargetSystem
)中。我们通过继承的方式来实现适配。
# 目标接口
class TargetSystem:def request(self):pass# 源接口
class OldSystem:def specific_request(self):return "Old system request"# 类适配器:继承源接口,适配成目标接口
class Adapter(TargetSystem, OldSystem):def request(self):# 调用源接口的方法,并将其适配为目标接口return self.specific_request()# 客户端代码
if __name__ == "__main__":target = Adapter()print(target.request()) # 输出:Old system request
解释:
TargetSystem
是客户端期望的目标接口。OldSystem
是现有的接口,它的方法是specific_request
,客户端不直接兼容。Adapter
继承了TargetSystem
和OldSystem
,并实现了目标接口的request
方法,将其适配为源接口的方法specific_request
。
6.2 示例:对象适配器模式
在对象适配器模式中,我们不通过继承,而是通过组合将源对象与目标接口适配。
# 目标接口
class TargetSystem:def request(self):pass# 源接口
class OldSystem:def specific_request(self):return "Old system request"# 对象适配器:通过组合的方式适配
class Adapter(TargetSystem):def __init__(self, old_system: OldSystem):self.old_system = old_system # 组合OldSystem实例def request(self):# 调用源接口的方法,将其适配为目标接口return self.old_system.specific_request()# 客户端代码
if __name__ == "__main__":old_system = OldSystem()target = Adapter(old_system)print(target.request()) # 输出:Old system request
解释:
TargetSystem
是目标接口,客户端期望调用的接口。OldSystem
是已有的源接口,无法直接满足目标接口。Adapter
类通过组合OldSystem
的实例来实现目标接口。
7. 适配器模式的总结
适配器模式是一个非常实用的设计模式,它通过将不兼容的接口转换为目标接口,帮助我们解决系统中不同模块或类之间接口不一致的问题。无论是类适配器模式还是对象适配器模式,都能够使得不同接口的类能够一起工作,提高代码的复用性、灵活性和扩展性。
适配器模式的核心优势:
- 解决接口不兼容问题。
- 保持原有系统的可扩展性和复用性。
- 降低系统之间的耦合度。
适配器模式的应用场景:
- 接口不兼容时需要进行转换。
- 在软件开发中集成第三方库时,常常需要进行接口适配。
- 系统需要统一多个接口时,适配器模式非常适用。
希望通过本文的介绍,能够帮助你更好地理解适配器模式,并能够在实际开发中灵活使用它。
相关文章:
设计模式-结构型-适配器模式
在软件开发中,随着系统的不断扩展和模块的不断增加,往往会遇到不同模块之间接口不兼容的情况。此时,如果我们能通过某种方式将一个接口转化为另一个接口,那么开发工作将变得更加灵活和高效。适配器模式(Adapter Patter…...
鸿蒙操作系统(HarmonyOS)
鸿蒙操作系统(HarmonyOS)是华为公司推出的一款面向未来、面向全场景的分布式操作系统。它旨在为用户提供一个更加智能、便捷和安全的操作环境,支持多种终端设备之间的无缝协作。在鸿蒙应用开发中,ArkUI作为官方推荐的用户界面开发…...

基于海思soc的智能产品开发(camera sensor的两种接口)
【 声明:版权所有,欢迎转载,请勿用于商业用途。 联系信箱:feixiaoxing 163.com】 对于嵌入式开发设备来说,除了图像显示,图像输入也是很重要的一部分。说到图像输入,就不得不提到camera。目前ca…...
解密LLM结构化输出:代码示例与原理分析
解密LLM结构化输出:代码示例与原理分析 一、LLM结构化输出概述 1. 结构化输出的定义与优势 结构化输出指的是语言模型(LLM)生成的遵循特定格式(如JSON、XML)的数据,这些数据易于解析和处理。相较于非结构…...
Go语言的数据类型
Go语言的数据类型详解 Go语言是一门具有简洁、高效并且强类型的编程语言。它的设计理念之一是让程序员能够以清晰、简明的方式表达自己的意图。在Go语言中,数据类型是其基础构建块之一,理解不同数据类型的特点和使用场景对于编写高效的Go程序至关重要。…...

复杂园区网基本分支的构建
目录 1、各主机进行网络配置。2、交换机配置。3、配置路由交换,进行测试。4、配置路由器接口和静态路由,进行测试。5、最后测试任意两台主机通信情况 模拟环境链接 拓扑结构 说明: VLAN标签在上面的一定是GigabitEthernet接口的,…...

如何很快将文件转换成另外一种编码格式?编码?按指定编码格式编译?如何检测文件编码格式?Java .class文件编码和JVM运行期内存编码?
如何很快将文件转换成另外一种编码格式? 利用VS Code右下角的"选择编码"功能,选择"通过编码保存"可以很方便将文件转换成另外一种编码格式。尤其,在测试w/ BOM或w/o BOM, 或者ANSI编码和UTF编码转换,特别方便。VS文件另…...

《C++11》Lambda 匿名函数从入门到进阶 优缺点分析 示例
Lambda 匿名函数从入门到进阶 C11 引入了 lambda 表达式,这是一种非常强大的功能,可以让我们在代码中定义匿名函数。它们不仅使代码更加简洁,而且在处理回调、算法和多线程编程时极为方便。本文将带你从入门到进阶,全面了解 C11 …...

连接Milvus
连接到Milvus 验证Milvus服务器正在侦听哪个本地端口。将容器名称替换为您自己的名称。 docker port milvus-standalone 19530/tcp docker port milvus-standalone 2379/tcp docker port milvus-standalone 192.168.1.242:9091/api/v1/health 使用浏览器访问连接地址htt…...
Linux——修改文件夹的所属用户组和用户
一、命令 举例: 授权 MOT17 文件夹 给 hust_xxx 用户: sudo chown -R hust_xxx:hust_xxx MOT17参考 Linux授权文件夹给用户...

Vue Amazing UI 组件库(Vue3+TypeScript+Vite 等最新技术栈开发)
Vue Amazing UI 一个 Vue 3 组件库 使用 TypeScript,都是单文件组件 (SFC),支持 tree shaking 有点意思 English | 中文 Vue Amazing UI 是一个基于 Vue 3、TypeScript、Vite 等最新技术栈开发构建的现代化组件库,包含丰富的 UI 组件和常…...

计算机Steam报错failedtoloadsteamui.dll怎么解决?DLL报错要怎么修复?
计算机Steam报错“Failed to Load SteamUI.dll”?这里有专业的解决方案! 作为软件开发领域的一名从业者,我深知电脑在运行过程中可能会遇到的各种问题,尤其是像Steam这样的大型游戏平台。今天,我将为大家科普一下Stea…...

如何开发一个简单的 dApp
后端合约 执行 sui move new resource_manage 创建一个包 接着就可以开始编写合约了 首先创建两个 Struct 用来创建 Profile 并记录在 State 中 public struct State has key {id: UID,users: Table<address, address>, }public struct Profile has key {id: UID,nam…...
TDengine 签约智园数字,助力化工园区智联未来
近年来,随着化工行业对安全、环保、高效运营的要求日益提高,化工园区的数字化转型成为必然趋势。从数据孤岛到全面互联,从基础监控到智能分析,如何高效管理和利用时序数据已成为化工园区智能化升级的关键环节。作为一家专注于时序…...

《Python游戏编程入门》注-第9章8
2 游戏信息的显示 在游戏窗口的上部会显示游戏分数、游戏关卡、剩余砖块数以及剩余小球数等信息,如图12所示。 图12 游戏信息显示 使用如图13所示的代码实现以上功能。 图13 显示游戏信息的代码 其中,print_text()函数MyLibrary....

js逆向实战(1)-- 某☁️音乐下载
下载某云音乐源文件.mp4格式 首先随便点进一首歌,如图所示获取该音乐id,然后点击播放键,打开F12进行查询XHR 由此可知,实际请求网址是 https://music.163.com/weapi/song/enhance/player/url/v1?csrf_token「你的token」url需带…...

AIA - APLIC之三(附APLIC处理流程图)
本文属于《 RISC-V指令集基础系列教程》之一,欢迎查看其它文章。 1 APLIC复位 APLIC复位后,其所有状态都变得有效且一致,但以下情况除外: 每个中断域的domaincfg寄存器(spec第 4.5.1 节);可能是machine-level interrupt domain的MSI地址配置寄存器(spec第4.5.3 和4.5…...

React Router 向路由组件传state参数浏览器回退历史页面显示效果问题
昨天在看尚硅谷张天禹老师讲的 React教程p90,老师讲到 React路由的 replace模式和push模式,老师的演示效果与自己本地操作不太一样。 老师的效果:点击查看消息1,消息2,消息3 再点回退,可以依次查看到 消息…...
线程池与并发工具:Java的分身管理器
1 线程池的概念 线程池是一种执行器(Executor),用于在一个后台线程中执行任务。线程池的主要目的是减少在创建和销毁线程时所产生的性能开销。通过重用已经创建的线程来执行新的任务,线程池提高了程序的响应速度,并且提…...
字玩FontPlayer开发笔记8 Tauri2文件系统
字玩FontPlayer开发笔记8 Tauri2文件系统 字玩FontPlayer是笔者开源的一款字体设计工具,使用Vue3 ElementUI开发,源代码: github: https://github.com/HiToysMaker/fontplayer gitee: https://gitee.com/toysmaker/fontplayer 笔记 字玩目…...

【Axure高保真原型】引导弹窗
今天和大家中分享引导弹窗的原型模板,载入页面后,会显示引导弹窗,适用于引导用户使用页面,点击完成后,会显示下一个引导弹窗,直至最后一个引导弹窗完成后进入首页。具体效果可以点击下方视频观看或打开下方…...

基于距离变化能量开销动态调整的WSN低功耗拓扑控制开销算法matlab仿真
目录 1.程序功能描述 2.测试软件版本以及运行结果展示 3.核心程序 4.算法仿真参数 5.算法理论概述 6.参考文献 7.完整程序 1.程序功能描述 通过动态调整节点通信的能量开销,平衡网络负载,延长WSN生命周期。具体通过建立基于距离的能量消耗模型&am…...

什么是库存周转?如何用进销存系统提高库存周转率?
你可能听说过这样一句话: “利润不是赚出来的,是管出来的。” 尤其是在制造业、批发零售、电商这类“货堆成山”的行业,很多企业看着销售不错,账上却没钱、利润也不见了,一翻库存才发现: 一堆卖不动的旧货…...
反射获取方法和属性
Java反射获取方法 在Java中,反射(Reflection)是一种强大的机制,允许程序在运行时访问和操作类的内部属性和方法。通过反射,可以动态地创建对象、调用方法、改变属性值,这在很多Java框架中如Spring和Hiberna…...

Netty从入门到进阶(二)
二、Netty入门 1. 概述 1.1 Netty是什么 Netty is an asynchronous event-driven network application framework for rapid development of maintainable high performance protocol servers & clients. Netty是一个异步的、基于事件驱动的网络应用框架,用于…...
【SpringBoot自动化部署】
SpringBoot自动化部署方法 使用Jenkins进行持续集成与部署 Jenkins是最常用的自动化部署工具之一,能够实现代码拉取、构建、测试和部署的全流程自动化。 配置Jenkins任务时,需要添加Git仓库地址和凭证,设置构建触发器(如GitHub…...
git: early EOF
macOS报错: Initialized empty Git repository in /usr/local/Homebrew/Library/Taps/homebrew/homebrew-core/.git/ remote: Enumerating objects: 2691797, done. remote: Counting objects: 100% (1760/1760), done. remote: Compressing objects: 100% (636/636…...
Python网页自动化Selenium中文文档
1. 安装 1.1. 安装 Selenium Python bindings 提供了一个简单的API,让你使用Selenium WebDriver来编写功能/校验测试。 通过Selenium Python的API,你可以非常直观的使用Selenium WebDriver的所有功能。 Selenium Python bindings 使用非常简洁方便的A…...

实战设计模式之模板方法模式
概述 模板方法模式定义了一个操作中的算法骨架,并将某些步骤延迟到子类中实现。模板方法使得子类可以在不改变算法结构的前提下,重新定义算法中的某些步骤。简单来说,就是在一个方法中定义了要执行的步骤顺序或算法框架,但允许子类…...
怎么开发一个网络协议模块(C语言框架)之(六) ——通用对象池总结(核心)
+---------------------------+ | operEntryTbl[] | ← 操作对象池 (对象数组) +---------------------------+ | 0 | 1 | 2 | ... | N-1 | +---------------------------+↓ 初始化时全部加入 +------------------------+ +-------------------------+ | …...