Java二十三种设计模式-适配器模式(6/23)

适配器模式:使不兼容的接口协同工作的桥梁

引言

适配器模式(Adapter Pattern)是一种结构型设计模式,它允许不兼容的接口之间可以一起工作,通过将一个类的接口转换成客户端期望的另一个接口。

在计算机编程中,适配器模式(有时候也称包装样式或者包装)将一个类的接口适配成用户所期待的。一个适配允许通常因为接口不兼容而不能在一起工作的类工作在一起,做法是将类自己的接口包裹在一个已存在的类中。

基础知识,java设计模式总体来说设计模式分为三大类:

(1)创建型模式,共5种:工厂方法模式、抽象工厂模式、单例模式、建造者模式、原型模式。

(2)结构型模式,共7种:适配器模式、装饰器模式、代理模式、外观模式、桥接模式、组合模式、享元模式。

(3)行为型模式,共11种:策略模式、模板方法模式、观察者模式、迭代子模式、责任链模式、命令模式、备忘录模式、状态模式、访问者模式、中介者模式、解释器模式。

第一部分:适配器模式概述

1.1 定义与目的

适配器模式的基本定义

适配器模式是一种结构型设计模式,其目的是通过一个中间层(适配器)将一个类的接口转换成客户端期望的另一个接口,从而使原本不兼容的接口能够一起工作。

解释为何需要适配器模式

  • 接口不兼容:当现有的类库提供的接口与消费者期望的接口不一致时,适配器模式提供了一种解决方案。
  • 重用现有类:适配器模式允许重用现有的类,即使这些类的接口不完全符合需求。
  • 解耦系统组件:适配器模式将接口转换逻辑与使用这些接口的客户端代码分离,降低了系统组件之间的耦合度。

1.2 适配器模式的类型

类适配器模式

  • 定义:通过继承和组合的方式,将一个类的接口转换成另一种形式。
  • 实现方式:创建一个新类(适配器类),继承目标接口,并在内部包含一个被适配者的实例。
  • 适用场景:当需要适配的类可以被继承,或者需要将多个不兼容的接口整合到一个统一的接口时。

对象适配器模式

  • 定义:通过对象组合的方式,让一个类的接口转换成客户期望的另一个接口。
  • 实现方式:创建一个新类(适配器类),实现目标接口,并通过引用包含一个被适配者的对象。
  • 适用场景:当需要适配的类不能被继承,或者想要避免使用继承时。

类适配器模式和对象适配器模式的对比

  • 继承与组合:类适配器模式使用继承实现,而对象适配器模式使用对象组合实现。
  • 灵活性:对象适配器模式提供了更高的灵活性,可以在运行时动态地更换适配器中的被适配者对象。
  • 复杂性:类适配器模式可能会增加系统的复杂性,因为它涉及到类的继承。

适配器模式通过提供一个中间层来转换接口,使得不兼容的接口能够协同工作。选择合适的适配器模式类型(类适配器或对象适配器)取决于是否需要继承被适配者以及对系统灵活性的需求。在下一部分中,我们将详细介绍适配器模式的组成与实现。

第二部分:适配器模式的组成与实现

2.1 角色定义

目标接口(Target Interface)

  • 定义:客户端所期望的接口。
  • 角色:定义客户端使用的特定领域相关的接口。

适配器类(Adapter Class)

  • 定义:适配器模式的核心,实现目标接口并包含对被适配者的引用。
  • 角色:通过在内部使用被适配者对象,把被适配者的接口转换成客户端期望的接口。

客户端(Client)

  • 角色:使用目标接口的类,不知道适配器内部如何工作的。
  • 职责:通过目标接口与适配器交互,达到与被适配者交互的目的。

被适配者(Adaptee)

  • 定义:一个已经存在的类,需要适配。
  • 角色:提供原始的操作和数据,但接口与客户端期望的接口不兼容。

2.2 Java实现示例

以下是使用Java语言实现适配器模式的示例。假设我们有一个不兼容的接口Adaptee,我们希望将其适配到客户端期望的Target接口。

// 目标接口
interface Target {void request();
}// 被适配者接口
class Adaptee {public void specificRequest() {System.out.println("Executing specific request.");}
}// 类适配器模式
class Adapter extends Adaptee implements Target {public void request() {specificRequest(); // 将被适配者的方法映射到目标接口}
}// 客户端代码
public class Client {public static void main(String[] args) {Target adapter = new Adapter();adapter.request();}
}

在这个示例中,Adapter类继承自Adaptee并实现了Target接口。request()方法将AdapteespecificRequest()方法映射到Target接口上,从而使得客户端可以通过Target接口与Adaptee对象交互。

对象适配器实现示例

// 目标接口
interface Target {void request();
}// 被适配者
class Adaptee {public void specificRequest() {System.out.println("Executing specific request.");}
}// 对象适配器模式
class ObjectAdapter implements Target {private Adaptee adaptee;public ObjectAdapter(Adaptee adaptee) {this.adaptee = adaptee;}public void request() {adaptee.specificRequest(); // 委托给被适配者}
}// 客户端代码
public class Client {public static void main(String[] args) {Adaptee adaptee = new Adaptee();Target adapter = new ObjectAdapter(adaptee);adapter.request();}
}

在这个对象适配器示例中,ObjectAdapter类实现了Target接口,并通过组合Adaptee对象来实现request()方法。这种方式提供了更大的灵活性,允许在运行时动态地更换被适配者对象。

适配器模式通过将一个类的接口转换成另一种形式,使得原本由于接口不兼容而不能一起工作的类可以协同工作。在下一部分中,我们将探讨适配器模式的使用场景。

第三部分:适配器模式的使用场景

3.1 系统间的接口不兼容

在软件开发过程中,经常会遇到需要集成不同系统的情况,这些系统可能因为设计时间、设计团队或者技术选型的差异而拥有不同的接口规范。

适配器模式如何发挥作用:

  • 统一接口:适配器模式通过提供一个统一的接口,使得不同系统的组件能够以一致的方式进行交互。
  • 减少修改:避免了对现有系统接口的修改,保护了现有系统的稳定性。
  • 简化集成:简化了不同系统间集成的复杂性,因为所有的交互都通过适配器进行。

应用实例:

  • 支付系统集成:在电子商务平台中,可能需要集成多种支付方式,如支付宝、微信支付等。每种支付方式都有自己特定的接口,通过适配器模式可以为平台提供一个统一的支付接口。

3.2 重用现有的类库

在许多情况下,开发者需要重用现有的类库,但这些类库的接口可能与当前系统的接口不兼容。

适配器模式的优势:

  • 提高复用性:适配器模式使得开发者能够重用现有的类库,即使这些类库的接口不完全符合当前系统的需求。
  • 解耦系统:适配器模式将接口转换逻辑与系统其他部分解耦,提高了系统的灵活性和可维护性。
  • 降低成本:通过适配器模式重用现有的类库,可以减少开发时间和成本。

应用实例:

  • 第三方API集成:在开发移动应用时,可能需要集成第三方API,如地图服务、社交媒体分享等。这些API可能具有不同的接口风格,适配器模式可以为应用提供一个统一的接口来访问这些服务。

适配器模式通过提供一个中间层来转换接口,使得不兼容的接口能够协同工作。它在系统间的接口不兼容和需要重用现有类库的场景下尤其有用。在下一部分中,我们将讨论适配器模式的优点与缺点。

第四部分:适配器模式的优点与缺点

4.1 优点

提高兼容性

  • 接口统一:适配器模式允许通过统一的接口与多种不同的接口进行交互,提高了不同系统或组件之间的兼容性。

复用性增强

  • 现有类库利用:适配器模式使得开发者能够复用现有的类库,即使这些库的接口与需求不完全匹配。
  • 减少代码冗余:通过适配器模式,避免了对不兼容接口的多次修改和适配,减少了代码冗余。

灵活性提升

  • 动态替换:在对象适配器模式中,可以在运行时动态地替换适配器中的被适配者对象,提高了系统的灵活性。

降低耦合度

  • 解耦系统组件:适配器模式将客户端与被适配者解耦,减少了它们之间的依赖关系。

4.2 缺点

增加系统的复杂性

  • 类或对象数量增加:适配器模式可能会增加系统中类或对象的数量,从而增加了系统的复杂性。

降低代码的透明度

  • 隐藏实现细节:适配器模式可能会隐藏被适配者的一些实现细节,使得客户端难以理解适配器的具体行为。

适配器职责过重

  • 适配器类膨胀:如果适配器承担了过多的适配职责,可能会导致适配器类过于庞大和复杂。

难以维护

  • 多个适配器维护:在系统中使用多个适配器时,可能难以维护和更新这些适配器。

性能考虑

  • 性能开销:适配器模式可能会引入额外的性能开销,尤其是在适配器执行复杂转换逻辑时。

适配器模式是一种强大的设计模式,可以提高系统的兼容性和复用性,同时提供灵活性和降低耦合度。然而,它也需要谨慎使用,以避免增加系统的复杂性和维护难度。在实际应用中,根据具体需求和场景选择是否使用适配器模式至关重要。在下一部分中,我们将比较适配器模式与其他设计模式,并提供一些最佳实践和建议。

第五部分:适配器模式与其他模式的比较

5.1 与装饰者模式的比较

装饰者模式

  • 定义:允许向一个现有的对象添加新的功能,同时又不改变其结构。装饰者提供了一种灵活的替代方案来扩展功能。
  • 使用场景:当你需要为对象动态地添加职责,但又不想用继承来实现时。

适配器模式

  • 定义:将一个类的接口转换成客户期望的另一个接口,使原本由于接口不兼容而不能一起工作的类可以协同工作。
  • 使用场景:当需要整合使用不同接口的类库时。

对比

  • 目的不同:装饰者模式的目的是增加对象的额外职责,而适配器模式的目的是使接口不兼容的对象能够一起工作。
  • 结构变化:装饰者模式通过组合可以动态地添加新的行为,而适配器模式通常在编译时就已经确定。
  • 使用方式:装饰者模式可以多层嵌套使用,形成装饰者链,而适配器模式通常只用于单个接口的转换。

5.2 与外观模式的对比

外观模式

  • 定义:提供了一个统一的接口,用来访问子系统中的一群接口,它定义了一个高层接口,让子系统更容易使用。
  • 使用场景:当需要简化复杂的系统或库的接口,提供简化的访问方式时。

适配器模式

  • 定义:允许不兼容的接口一起工作,通过创建一个中间层来转换接口。

对比

  • 接口简化:外观模式用于简化复杂系统的接口,而适配器模式用于解决接口不兼容的问题。
  • 目标不同:外观模式的目标是提供一个更简单的接口,隐藏内部的复杂性;适配器模式的目标是使得不同接口能够协同工作。
  • 使用范围:外观模式通常用于整个系统的简化,而适配器模式可以用于单个接口的转换或整个子系统的适配。

适配器模式、装饰者模式和外观模式都是常用的结构型设计模式,它们各自解决不同的设计问题。适配器模式专注于接口的兼容性,装饰者模式用于动态地添加职责,而外观模式用于简化复杂系统的接口。在实际应用中,根据具体的需求和场景选择合适的模式是非常重要的。在下一部分中,我们将提供适配器模式的最佳实践和建议。

第六部分:适配器模式的最佳实践和建议

6.1 最佳实践

保持适配器的简洁性

  • 单一职责:确保适配器只处理与接口转换相关的逻辑,避免添加额外的业务逻辑。

明确适配器的角色

  • 作为中介:适配器应该仅仅作为一个中介,不参与业务逻辑的实现。

适配器的扩展性

  • 预留扩展点:在设计适配器时,考虑未来可能的扩展,使适配器能够容易地适应新的接口变化。

适配器与被适配者的关系

  • 松耦合:适配器与被适配者之间应保持松耦合的关系,减少对被适配者内部实现的依赖。

适配器的测试

  • 独立测试:为适配器编写单元测试,确保其正确地转换接口。

6.2 避免滥用

避免过度耦合

  • 减少依赖:避免适配器对被适配者的过度依赖,这样当被适配者变化时,适配器不会受到太大影响。

避免适配器过于复杂

  • 单一功能:确保适配器专注于单一功能的适配,避免将多个适配逻辑混合在一个适配器中。

避免适配器的过度使用

  • 合理使用:适配器模式应该在确实需要接口转换时使用,而不是作为一种习惯性的设计手段。

6.3 替代方案

使用桥接模式

  • 定义:桥接模式将抽象部分与其实现部分分离,使它们可以独立地变化。
  • 适用场景:当需要将一个类族的功能和实现分离,以便独立地进行扩展时。

使用组合模式

  • 定义:组合模式允许将对象组合成树形结构以表示“部分-整体”的层次结构。
  • 适用场景:当需要处理的对象可以形成树形结构时。

使用继承

  • 适用场景:如果适配器模式用于解决继承问题,可以考虑使用继承来实现功能扩展。

使用外观模式

  • 适用场景:当需要简化复杂系统的接口时,可以考虑使用外观模式。

适配器模式是一种强大的设计模式,可以提高系统的兼容性和复用性。然而,合理使用适配器模式并避免滥用是至关重要的。了解其替代方案可以帮助开发者根据具体需求和场景选择最合适的设计模式。在实际开发中,应根据具体情况灵活运用适配器模式,以达到最佳的设计效果。

 

结语

适配器模式是一种在软件开发中常用的设计模式,它帮助解决了接口不兼容的问题,提高了代码的复用性。通过本文的深入分析,希望读者能够对适配器模式有更全面的理解,并在实际开发中做出合理的设计选择。

 

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

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

相关文章

C语言 ——— 调试的时候如何查看当前程序的变量信息

目录 调试前/后的调试窗口 ​编辑 调试窗口 --- 监视 调试窗口 --- 内存 调试窗口 --- 调用堆栈 调试前/后的调试窗口 调试前的调试窗口: 调试前的调试窗口是没有显示的,只有在调试的时候才会有相对应的调试窗口 调试后的调试窗口&#xff1a…

【JVM基础01】——介绍-初识JVM运行流程

目录 1- 引言:初识JVM1-1 JVM是什么?(What)1-1-1 概念1-1-2 优点 1-2 为什么学习JVM?(Why) 2- 核心:JVM工作的原理(How)⭐2-1 JVM 的组成部分及工作流程2-2 学习侧重点 3- 小结(知识点大纲):3-1 JVM 组成3…

风险评估:IIS的安全配置,IIS安全基线检查加固

「作者简介」:冬奥会网络安全中国代表队,CSDN Top100,就职奇安信多年,以实战工作为基础著作 《网络安全自学教程》,适合基础薄弱的同学系统化的学习网络安全,用最短的时间掌握最核心的技术。 这一章节我们需…

MySQL运维实战之ProxySQL(9.9)proxysql自身高可用

作者:俊达 proxysql作为一个程序,本身也可能出现故障。部署proxysql的服务器也肯能出现故障。高可用架构的一个基本原则是消除单点。 可以在多个节点上部署proxysql,在proxysql之前再加一层负载均衡(如使用LVS或其他技术&#x…

Ubuntu 22.04.4 LTS (linux) 安装certbot 免费ssl证书申请 letsencrypt

1 安装certbot sudo apt update sudo apt-get install certbot 2 申请letsencrypt证书 sudo certbot certonly --webroot -w 网站目录 -d daloradius.域名.com 3 修改nginx 配置ssl 证书 # 配置服务器证书 ssl_certificate /etc/letsencrypt/live/daloradius.域名.com/f…

Redis的热key解决

1、Redis热Key会带来哪些问题 1、流量集中,达到物理网卡上限。 当某一热点 Key 的请求在某一主机上超过该主机网卡上限时,由于流量的过度集中,会导致服务器中其它服务无法进行。 2、请求过多,缓存分片服务被打垮。 如果热点过于…

AI绘画入门实践|Midjourney 提示词的使用技巧

提示词长短 尽可能做到简洁明了。 提示词很短 MJ 出图的随机性更高,创造的内容更有想象力,更适合创意发散的图像生成。 a dog 提示词很长 MJ 出图会更加精准,但描述太过详细,有可能出现AI理解不到位的情况。 越到后面的提示词&…

17-8 向量数据库之野望8 - 7 个主流向量数据库

​​​​​​ 在快速发展的人工智能 (AI)、机器学习 (ML) 和数据工程领域,对高效数据存储和检索系统的需求至关重要。矢量数据库已成为管理这些技术通常依赖的复杂高维数据的关键解决方案。在这里,我们探讨了每个 AI/ML/数据工程师都应该熟悉的七个矢量数据库,重点介绍了它们…

智能物流锁控如何重塑货运物流货物追踪与监控

一、物流智能锁控行业现状分析 1.1 传统锁控系统局限性 安全性不足:机械锁容易被撬开或钥匙被复制,导致货物在运输过程中面临被盗风险。据统计,每年因货物丢失或损坏导致的经济损失高达数十亿美元。 无法实时追踪:传统锁控系统…

Linux下Qt程序打包

文章目录 一、前言二、linuxdeployqt下载安装三、Qt环境变量配置四、准备Qt可执行文件五、打包 一、前言 在Windows下进行Qt开发,软件开发好之后可以使用windeployqt进行打包,然后程序就可以移动到其它电脑上运行了 在Linux下同样可以进行打包&#xf…

7.SpringBoot整合Neo4j

1.引入依赖 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-neo4j</artifactId> </dependency> 说明&#xff1a;这里引入neo4j的版本跟spring框架的版本有关系。需要注意不同的版本在neo…

【java】力扣 买卖股票的最佳时机 动态规划

文章目录 题目链接题目描述思路代码 题目链接 121.买卖股票的最佳时机 题目描述 思路 本题主要用到了动态规划 1.先定义dp数组的含义 先定义一个二维数组dp 然后dp[i][0]来表示第i天持有股票的现金 dp[i][1]代表第i天不持有股票的现金 刚开始的现金为0&#xff0c;当第i天买…

CentOS 7 Web面板的文件管理器说明

在使用CentOS 7 Web Panel&#xff08;CWP7&#xff09;时&#xff0c;偶尔要求在服务器曲面上修改&#xff0c;创建&#xff0c;编辑或删除文件。 最简单&#xff0c;最直接的方式是通过利用CWP7的内置文件管理器。 本文将详细介绍如何启动它&#xff0c;使用它&#xff0c;以…

利用OSMnx进行城市路网数据的速度与通行时间推算及分析

本文还是以广州市路网为例&#xff0c;通过osmmx调用ox.add_edge_speeds(G)时&#xff0c;该函数会遍历图G 中的每条边&#xff08;即每条街道&#xff09;&#xff0c;并基于一些预设的规则或街道属性&#xff08;如街道类型、是否为主要道路、是否有速度限制等&#xff09;来…

Golang | Leetcode Golang题解之第235题二叉搜索树的最近公共祖先

题目&#xff1a; 题解&#xff1a; func lowestCommonAncestor(root, p, q *TreeNode) (ancestor *TreeNode) {ancestor rootfor {if p.Val < ancestor.Val && q.Val < ancestor.Val {ancestor ancestor.Left} else if p.Val > ancestor.Val && q…

十五、【文本编辑器(一)】代码框架

目录 一、QMainWindow基本构成 二、项目框架代码 三、菜单与工具栏的实现 四、总结 一、QMainWindow基本构成 QMainWindow 是一个为用户提供主窗口程序的类&#xff0c;包含一个菜单栏 (menu bar)&#xff0c;多个工具栏(tool bars)&#xff0c;多个锚接部件 (dock widgets…

最新版康泰克完整版- Kontakt v7.10.5 for Win和Mac,支持m芯片和intel,有入库工具

一。世界最受欢迎的采样器的新篇章 Native Instruments Kontakt是采样器领域的标准&#xff0c;您将获得高质量的滤波器&#xff0c;在这里您将找到经典的模拟电路和最现代的滤波器。每一个都可以根据您的口味进行定制&#xff0c;并且由于它&#xff0c;您可以获得前所未有的声…

debian12 允许远程ssh登录

1、查看ssh服务是否在运行 systemctl status ssh 如果没有安装ssh服务&#xff0c;可以通过以下命令安装&#xff1a; apt install openssh-server 如果服务没有运行&#xff0c;通过以下命令重启ssh服务&#xff1a; systemctl restart ssh 2、编辑配置文件 vim /etc/ss…

DROO论文笔记

推荐文章DROO源码及论文学习 读论文《Deep Reinforcement Learning for Online Computation Offloading in Wireless Powered Mobile-Edge Computing Networks》的笔记 论文地址&#xff1a;用于无线移动边缘计算网络在线计算卸载的深度强化学习 论文代码地址&#xff1a;DR…

数组去重(去重+单调栈 用boolen instack来去重

如果存到一个小的就把比它大的都pop出去&#xff0c;在后面还有的情况下&#xff0c;如果后面没有就不pop了 class Solution {public String removeDuplicateLetters(String s) {Stack<Character> stacknew Stack<>();int[] countnew int[256];boolean[] inStackn…