当前位置: 首页 > news >正文

设计模式之适配器模式:软件世界的桥梁建筑师

一、什么是适配器模式

    适配器模式(Adapter Pattern)是一种结构型设计模式(Structural Pattern),通过将类的接口转换为客户期望的另一个接口,适配器可以让不兼容的两个类一起协同工作。其核心思想是通过一个中间的“适配器”类,将一个类的接口转换成客户端所期待的另一种接口形式,从而使得原本因接口不兼容而不能一起工作的类能够协同工作。就像是在现实生活中,国际旅行中常用的电源转换器,就是将不同国家的插座标准转换为你的电子设备所支持的充电接口,使设备得以顺利充电。

二、适配器模式的结构

    适配器模式通常包含以下几个主要角色:

  1. 目标接口(Target):这是客户端所期待的接口,即客户端通过此接口来调用所需的业务逻辑。

  2. 适配者(Adaptee):需要适配的类或接口,通常其现有的接口与客户端所需的接口不兼容。

  3. 适配器(Adapter):这是适配器模式的核心,它继承适配者或者持有一个适配者的引用,并通过实现目标接口,将适配者的接口转换为客户端所期望的接口。

三、适配器模式的分类

    根据适配器如何适配适配者,适配器模式可以分为两种类型:类适配器模式和对象适配器模式

1. 类适配器模式

    类适配器通过实现目标接口的同时继承适配者的方式,来实现在内部使用适配者的方法来实现目标接口的的方法:

2. 对象适配器模式

    对象适配器模式则是通过组合的方式来实现接口的适配。适配器类持有一个适配者的引用,并通过该引用来调用源角色的方法,从而实现目标接口的方法。

四、适配器模式的应用场景

    适配器模式广泛应用于各种软件开发场景中,尤其是当面临不同系统、不同库或不同版本之间的接口不兼容问题时,适配器模式往往能够提供优雅的解决方案。以下是一些典型的应用场景:

  1. 旧系统迁移:在将旧系统迁移到新系统时,新系统可能不支持旧系统的某些接口或方法。此时,可以通过适配器模式将旧系统的接口适配为新系统所需的接口,从而保持旧系统功能的可用性。

  2. 第三方库集成:在集成第三方库时,如果该库的接口与我们的系统接口不兼容,可以通过适配器模式来封装第三方库的接口,使其符合我们的系统架构和接口规范。

  3. 接口升级:在软件系统的演进过程中,有时需要对接口进行升级或重构,但升级后的接口可能与旧客户端不兼容。此时,可以通过适配器模式为旧客户端提供一个兼容层,使它们能够继续访问系统而不必立即进行更新。

  4. 多平台支持:在开发跨平台应用程序时,不同平台可能提供不同的API接口。通过适配器模式,可以将各个平台的API适配为统一的接口,从而简化开发工作和代码维护。

五、适配器模式示例

    为了更直观地理解适配器模式,以下通过一个简单的示例来说明其实现过程。

     假设我们有一个音频播放器系统,该系统支持多种音频格式的播放,但现在我们需要接入一个新的音频设备,该设备只支持MP3格式的音频。然而,我们的音频库中存在一些非MP3格式的音频文件(如WAV格式),我们需要将这些非MP3格式的音频文件转换为MP3格式后才能在该设备上播放。

1. 定义目标接口

    首先,假设存在一个目标接口AudioPlayer,表示音频播放器的功能接口。

public interface AudioPlayer {void play(String audioType, String fileName);
}

2. 定义适配者

    另外,存在一个适配者Mp3Player,它代表了一个只支持MP3格式的播放器。

public class Mp3Player {public void playMp3(String fileName) {System.out.println("Playing mp3 file. Name: " + fileName);}
}

    注意:这里Mp3Player并没有直接提供play方法,而是提供了针对特定格式的播放方法,如playMp3。

3. 创建适配器类

    现在,我们需要创建一个适配器类AudioAdapter,它实现了AudioPlayer接口,并持有一个Mp3Player的实例,以便将非MP3格式的音频文件转换为MP3格式(在这个例子中,为了简化,我们假设转换逻辑已经内置在AudioAdapter中,实际上可能需要外部转换工具或服务)。

    然而,为了保持示例的简洁性和集中讨论适配器模式的核心思想,我们将省略实际的转换逻辑,而是直接调用相应的播放方法,并输出一个假设的转换过程。

public class AudioAdapter implements AudioPlayer {private final Mp3Player mp3Player;
​public AudioAdapter(Mp3Player mp3Player) {this.mp3Player = mp3Player;}
​@Overridepublic void play(String audioType, String fileName) {if ("wma".equalsIgnoreCase(audioType)) {// 假设这里进行了wma到MP3的转换System.out.println("Converting wma to mp3 format...");} else if ("wav".equalsIgnoreCase(audioType)) {// 假设这里进行了wav到MP3的转换System.out.println("Converting wav to mp3 format...");}mp3Player.playMp3(fileName);}
}

    请注意,上述代码中的play方法通过检查audioType参数来确定要播放的音频格式,并假设进行了一些格式转换。这里的关键点是适配器类AudioAdapter如何将一个不直接支持的接口(在这个例子中是play方法,其期望接收音频类型和文件名)适配为另一个接口(Mp3Player中的playMp3方法)。

4. 使用适配器

    最后,我们可以创建一个测试类来模拟使用这个适配器。

public class AudioPlayerTest {public static void main(String[] args) {AudioPlayer audioPlayer = new AudioAdapter(new Mp3Player());audioPlayer.play("mp3", "music.mp3");audioPlayer.play("wma", "music.wma");audioPlayer.play("wav", "music.wav");}
}

5. 运行结果

    在上面的例子中,我们创建了一个AudioPlayer类型的对象,但实际上它是一个AudioAdapter的实例,该实例内部封装了一个Mp3Player对象。通过AudioAdapter,我们能够以AudioPlayer接口期望的方式(即调用play方法并传入音频类型和文件名)来播放非MP3格式的音频文件,尽管AdvancedAudioPlayer本身并不支持这种调用方式。

相关文章:

设计模式之适配器模式:软件世界的桥梁建筑师

一、什么是适配器模式 适配器模式(Adapter Pattern)是一种结构型设计模式(Structural Pattern),通过将类的接口转换为客户期望的另一个接口,适配器可以让不兼容的两个类一起协同工作。其核心思想是通过一个…...

Java 入门指南:Java 并发编程 —— Fork/Join 框架 实现任务的拆分与合并

Fork/Join Fork/Join 是Java并发编程中的一个框架,用于解决大型任务的并行执行问题。它于 Java 7中引入,旨在简化对多核处理器上可并行执行任务的开发。 Fork/Join 框架基于分治(divide and conquer)的设计思想。它将大型任务划…...

token过期时间分平台(web和app)设置方法

token分平台设置方法 本文介绍了Spring下的登录和鉴权机制的主要方法以及 token认证的主要流程,并介绍在spring中web端和APP端设置不同token过期时间的实现方法。主要基于SpringBootspringSecurityJWT框架实现。 一、应用场景 同一系统的跨平台操作,基于…...

[000-01-008].Seata案例应用

业务说明:这里我们创建三个服务,一个订单服务,一个库存服务,一个账户服务。当用户下单时,会在订单服务中创建一个订单,然后通过远程调用库存服务来扣减下单商品的库存;再通过远程调用账户服务来…...

超详细!!!electron-vite-vue开发桌面应用之创建新窗口以及主进程和子进程的通信监听(十二)

云风网 云风笔记 云风知识库 一、新建打开窗口 1、在electron/main.ts中加入主进程打开窗口逻辑代码 import { ipcMain } from "electron"; ipcMain.handle("open-win", (_, arg) > {const childWindow new BrowserWindow({webPreferences: {preloa…...

java编辑器——IntelliJ IDEA

java编辑器有两种选择——IntelliJ IDEA和VsCode。其中IntelliJ IDEA现在是企业用的比较多的,是专门为java设计的,而VsCode则是通过插件来实现Java编辑的。 1.IntelliJ IDEA 官网下载链接:https://www.jetbrains.com/idea/ 注意选择社区版…...

经验笔记:SSL证书

SSL证书经验笔记 1. 什么是SSL证书? SSL(Secure Sockets Layer)证书是一种数字证书,用于在客户端(如浏览器)和服务器之间建立加密连接,以确保数据传输的安全性。随着互联网的发展,…...

设计模式之装饰器模式:让对象功能扩展更优雅的艺术

一、什么是装饰器模式 装饰器模式(Decorator Pattern)是一种结构型设计模式(Structural Pattern),它允许用户通过一种灵活的方式来动态地给一个对象添加一些额外的职责。就增加功能来说,装饰器模式相比使用…...

Anchor Alignment Metric来优化目标检测的标签分配和损失函数。

文章目录 背景假设情况任务和目标TaskAligned方法的应用1. **计算Anchor Alignment Metric**2. **动态样本分配**3. **调整损失函数** 示例总结 背景 假设我们在进行目标检测任务,并且使用了YOLOv8模型。我们希望通过TaskAligned方法来优化Anchor与目标的匹配程度&…...

C++---由优先级队列认识仿函数

文章目录 一、优先级队列是什么? 二、如何使用优先级队列 1、优先级队列容器用法 2、为什么容器本身无序? 三、什么是仿函数? 1. 什么是仿函数? 2. 仿函数的优势 四、仿函数如何使用? 1、重载operator()函数 2、运用第…...

Client访问Server访问慢的原因

1. 网络层面的问题 网络延迟:客户端与服务器之间的地理距离较远(跨ISP、路径次优),导致高网络延迟(如高 RTT 值)。使用 ping 或 traceroute 工具可以帮助定位网络延迟的来源 - mtr: 结合了ping和traceroute功能&#…...

用RPC Performance Inspector 优化你的区块链

目录 什么是RPC? RPC Performance Inspector 是做什么的? 为什么需要这个工具? 如何使用它? 适合谁用? 如何使用? 什么是RPC? RPC Performance Inspector 是一个专门用于测试和分析RPC性能…...

linux如何查看内存条是ddr几代

在 Linux 系统中,可以通过以下几种方法查看内存条的类型和代数(如 DDR3、DDR4 等): 1. 使用 dmidecode 命令 dmidecode 是一个工具,它可以从系统的 DMI 表(也称为 SMBIOS 表)中提取硬件信息&a…...

LeetCode 3153.所有数对中数位差之和:计数

【LetMeFly】3153.所有数对中数位差之和:计数 力扣题目链接:https://leetcode.cn/problems/sum-of-digit-differences-of-all-pairs/ 车尔尼有一个数组 nums ,它只包含 正 整数,所有正整数的数位长度都 相同 。 两个整数的 数位…...

Spring Boot 整合 Sentinel 实现流量控制

在微服务架构中,流量控制是保障系统稳定性和高可用性的关键技术之一。阿里巴巴开源的 Sentinel 是一款面向分布式系统的流量防护组件,旨在从流量控制、熔断降级、系统负载保护等多个维度保障服务的稳定性。本文将详细介绍如何在 Spring Boot 项目中整合 …...

Elasticsearch倒排索引

什么是倒排索引 倒排索引(Inverted Index)是一种将文档中的每个单词映射到包含该单词的文档列表上的数据结构 倒排索引的构建过程 文档1: “我爱吃苹果” 文档2: “我爱吃香蕉” 文档3: “我喜欢苹果和香蕉” 文档分词:将文档中的文本内容…...

速盾:ddos常用防御方法是什么?

DDoS攻击是一种通过向网络资源发送大量请求或大流量数据来使其过载的攻击手段。为了应对这种攻击,常用的防御方法可以分为三个层次:流量清洗、服务器升级和高防CDN。 流量清洗是一种基础的防御手段,它通过过滤和识别恶意流量来阻止DDoS攻击。…...

二分算法入门(简单题)

习题1 704. 二分查找 给定一个 n 个元素有序的(升序)整型数组 nums 和一个目标值 target ,写一个函数搜索 nums 中的 target,如果目标值存在返回下标,否则返回 -1。 示例 1: 输入: nums [-1,0,3,5,9,12], targ…...

在使用React Hooks中,如何避免状态更新时的性能问题?

在React Hooks中避免状态更新时的性能问题,可以采取以下一些最佳实践: 避免不必要的状态更新: 使用React.memo、useMemo、和useCallback来避免组件或其子组件进行不必要的渲染。 使用useMemo: 对于基于状态或props的复杂计算&…...

Pytest插件pytest-selenium-让自动化测试更简洁

在现代Web应用的开发中,自动化测试成为确保网站质量的重要手段之一。而Pytest插件 pytest-selenium 则为开发者提供了简单而强大的工具,以便于使用Python进行Web应用的自动化测试。本文将深入介绍 pytest-selenium 插件的基本用法和实际案例,…...

uniapp 对接腾讯云IM群组成员管理(增删改查)

UniApp 实战:腾讯云IM群组成员管理(增删改查) 一、前言 在社交类App开发中,群组成员管理是核心功能之一。本文将基于UniApp框架,结合腾讯云IM SDK,详细讲解如何实现群组成员的增删改查全流程。 权限校验…...

3.3.1_1 检错编码(奇偶校验码)

从这节课开始,我们会探讨数据链路层的差错控制功能,差错控制功能的主要目标是要发现并且解决一个帧内部的位错误,我们需要使用特殊的编码技术去发现帧内部的位错误,当我们发现位错误之后,通常来说有两种解决方案。第一…...

大型活动交通拥堵治理的视觉算法应用

大型活动下智慧交通的视觉分析应用 一、背景与挑战 大型活动(如演唱会、马拉松赛事、高考中考等)期间,城市交通面临瞬时人流车流激增、传统摄像头模糊、交通拥堵识别滞后等问题。以演唱会为例,暖城商圈曾因观众集中离场导致周边…...

【Redis技术进阶之路】「原理分析系列开篇」分析客户端和服务端网络诵信交互实现(服务端执行命令请求的过程 - 初始化服务器)

服务端执行命令请求的过程 【专栏简介】【技术大纲】【专栏目标】【目标人群】1. Redis爱好者与社区成员2. 后端开发和系统架构师3. 计算机专业的本科生及研究生 初始化服务器1. 初始化服务器状态结构初始化RedisServer变量 2. 加载相关系统配置和用户配置参数定制化配置参数案…...

渲染学进阶内容——模型

最近在写模组的时候发现渲染器里面离不开模型的定义,在渲染的第二篇文章中简单的讲解了一下关于模型部分的内容,其实不管是方块还是方块实体,都离不开模型的内容 🧱 一、CubeListBuilder 功能解析 CubeListBuilder 是 Minecraft Java 版模型系统的核心构建器,用于动态创…...

VTK如何让部分单位不可见

最近遇到一个需求&#xff0c;需要让一个vtkDataSet中的部分单元不可见&#xff0c;查阅了一些资料大概有以下几种方式 1.通过颜色映射表来进行&#xff0c;是最正规的做法 vtkNew<vtkLookupTable> lut; //值为0不显示&#xff0c;主要是最后一个参数&#xff0c;透明度…...

零基础设计模式——行为型模式 - 责任链模式

第四部分&#xff1a;行为型模式 - 责任链模式 (Chain of Responsibility Pattern) 欢迎来到行为型模式的学习&#xff01;行为型模式关注对象之间的职责分配、算法封装和对象间的交互。我们将学习的第一个行为型模式是责任链模式。 核心思想&#xff1a;使多个对象都有机会处…...

Android15默认授权浮窗权限

我们经常有那种需求&#xff0c;客户需要定制的apk集成在ROM中&#xff0c;并且默认授予其【显示在其他应用的上层】权限&#xff0c;也就是我们常说的浮窗权限&#xff0c;那么我们就可以通过以下方法在wms、ams等系统服务的systemReady()方法中调用即可实现预置应用默认授权浮…...

聊一聊接口测试的意义有哪些?

目录 一、隔离性 & 早期测试 二、保障系统集成质量 三、验证业务逻辑的核心层 四、提升测试效率与覆盖度 五、系统稳定性的守护者 六、驱动团队协作与契约管理 七、性能与扩展性的前置评估 八、持续交付的核心支撑 接口测试的意义可以从四个维度展开&#xff0c;首…...

3-11单元格区域边界定位(End属性)学习笔记

返回一个Range 对象&#xff0c;只读。该对象代表包含源区域的区域上端下端左端右端的最后一个单元格。等同于按键 End 向上键(End(xlUp))、End向下键(End(xlDown))、End向左键(End(xlToLeft)End向右键(End(xlToRight)) 注意&#xff1a;它移动的位置必须是相连的有内容的单元格…...