设计模式-结构型-适配器模式
在软件开发中,随着系统的不断扩展和模块的不断增加,往往会遇到不同模块之间接口不兼容的情况。此时,如果我们能通过某种方式将一个接口转化为另一个接口,那么开发工作将变得更加灵活和高效。适配器模式(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 笔记 字玩目…...
[特殊字符] 智能合约中的数据是如何在区块链中保持一致的?
🧠 智能合约中的数据是如何在区块链中保持一致的? 为什么所有区块链节点都能得出相同结果?合约调用这么复杂,状态真能保持一致吗?本篇带你从底层视角理解“状态一致性”的真相。 一、智能合约的数据存储在哪里…...
Golang 面试经典题:map 的 key 可以是什么类型?哪些不可以?
Golang 面试经典题:map 的 key 可以是什么类型?哪些不可以? 在 Golang 的面试中,map 类型的使用是一个常见的考点,其中对 key 类型的合法性 是一道常被提及的基础却很容易被忽视的问题。本文将带你深入理解 Golang 中…...
Day131 | 灵神 | 回溯算法 | 子集型 子集
Day131 | 灵神 | 回溯算法 | 子集型 子集 78.子集 78. 子集 - 力扣(LeetCode) 思路: 笔者写过很多次这道题了,不想写题解了,大家看灵神讲解吧 回溯算法套路①子集型回溯【基础算法精讲 14】_哔哩哔哩_bilibili 完…...
《Playwright:微软的自动化测试工具详解》
Playwright 简介:声明内容来自网络,将内容拼接整理出来的文档 Playwright 是微软开发的自动化测试工具,支持 Chrome、Firefox、Safari 等主流浏览器,提供多语言 API(Python、JavaScript、Java、.NET)。它的特点包括&a…...
JUC笔记(上)-复习 涉及死锁 volatile synchronized CAS 原子操作
一、上下文切换 即使单核CPU也可以进行多线程执行代码,CPU会给每个线程分配CPU时间片来实现这个机制。时间片非常短,所以CPU会不断地切换线程执行,从而让我们感觉多个线程是同时执行的。时间片一般是十几毫秒(ms)。通过时间片分配算法执行。…...
【开发技术】.Net使用FFmpeg视频特定帧上绘制内容
目录 一、目的 二、解决方案 2.1 什么是FFmpeg 2.2 FFmpeg主要功能 2.3 使用Xabe.FFmpeg调用FFmpeg功能 2.4 使用 FFmpeg 的 drawbox 滤镜来绘制 ROI 三、总结 一、目的 当前市场上有很多目标检测智能识别的相关算法,当前调用一个医疗行业的AI识别算法后返回…...
蓝桥杯3498 01串的熵
问题描述 对于一个长度为 23333333的 01 串, 如果其信息熵为 11625907.5798, 且 0 出现次数比 1 少, 那么这个 01 串中 0 出现了多少次? #include<iostream> #include<cmath> using namespace std;int n 23333333;int main() {//枚举 0 出现的次数//因…...
laravel8+vue3.0+element-plus搭建方法
创建 laravel8 项目 composer create-project --prefer-dist laravel/laravel laravel8 8.* 安装 laravel/ui composer require laravel/ui 修改 package.json 文件 "devDependencies": {"vue/compiler-sfc": "^3.0.7","axios": …...
搭建DNS域名解析服务器(正向解析资源文件)
正向解析资源文件 1)准备工作 服务端及客户端都关闭安全软件 [rootlocalhost ~]# systemctl stop firewalld [rootlocalhost ~]# setenforce 0 2)服务端安装软件:bind 1.配置yum源 [rootlocalhost ~]# cat /etc/yum.repos.d/base.repo [Base…...
嵌入式学习之系统编程(九)OSI模型、TCP/IP模型、UDP协议网络相关编程(6.3)
目录 一、网络编程--OSI模型 二、网络编程--TCP/IP模型 三、网络接口 四、UDP网络相关编程及主要函数 编辑编辑 UDP的特征 socke函数 bind函数 recvfrom函数(接收函数) sendto函数(发送函数) 五、网络编程之 UDP 用…...
