适配器模式 Adapter Pattern
https://en.wikipedia.org/wiki/Adapter_pattern
https://www.baeldung.com/java-adapter-pattern
适配器模式(也称为包装器「wrapper」,与装饰器模式「decorator pattern」共享的另一种命名),它允许将现有类的接口用作另一个接口。它通常用于使现有类与其他类协同工作,而无需修改其源代码。
适配器模式描述了如何解决重复出现的「recurring」设计问题,以设计灵活和可重用的面向对象软件,即更容易实现、更改、测试和重用的对象。
解决了以下问题:
如何重用没有客户端所需接口的类?
具有不兼容「incompatible」接口的类如何协同工作?
如何为类提供替代接口「alternative interface」?
通常,一个(已经存在的)类不能被重用,只是因为它的接口不符合「conform to」客户端要求的接口。
如何解决这些问题:
定义一个单独的适配器类「adapter」,将类(adaptee)的(不兼容)接口转换为客户端所需的另一个接口(target)。
通过适配器「adapter」处理(重用)没有所需接口的类。
这种模式的关键思想是通过一个单独的适配器「adapter」来工作,该适配器在不更改的情况下调整(现有)类的接口。
客户端不知道他们是直接使用目标类,还是通过适配器使用没有目标接口的类。
另请参见下面的UML类图。

在上面的UML类图中,需要 target 接口的 client 类不能直接重用adaptee类,因为 adaptee 的接口不符合target 接口。相反,客户端通过一个 adapter 类工作,该类根据adaptee实现了target 接口:
对象适配器「object adapter 」方式通过在运行时委托给 adaptee 对象来实现 target 接口(adaptee.specificOperation())。
类适配器「class adapter」方式通过在编译时继承 adaptee 类来实现目标接口(specificOperation())。
Object adapter pattern
在这个适配器模式中,适配器包含它所包装的类的一个实例。在这种情况下,适配器会调用包装对象的实例。

下面这个图片描述了 Adapter 实现了 Target,包含一个 Adaptee 的引用。

Class adapter pattern
此适配器模式使用多个多态接口「polymorphic interfacesmultiple polymorphic interfaces」,实现或继承预期的接口和预先存在的接口。通常将预期的接口创建为纯接口类,特别是在Java(JDK 1.8之前)等不支持类多重继承的语言中。

这了这个没有展示出,预期的接口 ,只展示了 预先存在的接口 Adaptee1-N。
所以来看看下图的解释,实现了(绿色箭头)预期的接口 ,继承了(蓝色箭头)预先存在的接口
继承是因为不一定对 adaptee 有控制权。

OOP中的类之间关系-CSDN博客
Benefits and Trade-Offs
类适配器方法最适合Target和Adaptee方法之间的一对一映射。这样,我们就可以使用委托,而无需在Adapter中进行额外的实现。但是,如果Target接口更复杂,这种方法可能需要在Adapter中进行额外的工作。然而,我们可以通过委托「delegation」来解决这个问题:

在这里,我们只将request()方法委托给Adaptee。其余部分取自ConcreteTarget。我们可以使用组合将这些接口方法委托给实现,以避免代码重复。同时,如果我们不需要双边「two-way」适配器,我们可以使用对象适配器,这将使结构更简单:
因此,实现此模式的方式在很大程度上取决于代码库的初始状态,我们是否可以使用接口,以及我们是否需要为适配器提供在两种情况下工作的能力。
Adapter Pattern Example
Java有一个很好的适配器模式示例,我们可以在这里查看。Enumeration 和 Iterator 是两个相关的接口,是adapter-adaptee 关系的很好的例子。
public interface Enumeration<E> {boolean hasMoreElements();E nextElement();default Iterator<E> asIterator() {return new Iterator<>() {@Override public boolean hasNext() {return hasMoreElements();}@Override public E next() {return nextElement();}};}}
Iterator接口的描述包含以下内容:
iterator 用来遍历 collection。iterator在Java集合框架中取代了Enumeration。Iterators 与Enumerations 有两个不同之处:
Iterators 允许调用者在迭代过程中使用定义良好的语义从底层集合中删除元素。
方法名称已得到改进。
从技术上讲,枚举具有相同的接口,唯一的区别是方法名称:
public interface Iterator<E> {boolean hasNext();E next();default void remove() {throw new UnsupportedOperationException("remove");}default void forEachRemaining(Consumer<? super E> action) {Objects.requireNonNull(action);while (hasNext())action.accept(next());}}
Adapter Implementations
正如我们所看到的,这些接口是相似的,具有相同的目标。默认的asIterator()方法是在Java 9中添加的,它包含使用匿名类实现Adapter模式:
default Iterator<E> asIterator() {return new Iterator<>() {@Override public boolean hasNext() {return hasMoreElements();}@Override public E next() {return nextElement();}};
}
这个例子使用了组合,但在这种情况下并不明确。我们不将枚举实例传递给迭代器,因为我们在枚举的上下文中创建了迭代器。这样,我们就可以直接访问Enumeration方法。这是一种非常强大的技术,它允许隐藏接口的一部分并使用委托给私有方法。前面的类和对象适配器示例需要公共API进行委派。
然而,只有当我们同时控制adapter 和 adaptee 时,才有可能使用匿名类实现Adapter模式,而这在大多数情况下是不可能的。让我们想象一下,在Java 9之前,我们如何实现相同的功能:
public class IteratorAdapter<E> implements Iterator<E> {private Enumeration<E> enumeration;public IteratorAdapter(Enumeration<E> enumeration) {this.enumeration = enumeration;}@Overridepublic boolean hasNext() {return enumeration.hasMoreElements();}@Overridepublic E next() {return enumeration.nextElement();}}
此示例与我们之前回顾的对象适配器示例相同。让我们用类适配器实现相同的功能。我们将在这个例子中使用StringTokenizer,因为它实现了枚举接口:
public class StringTokenizer implements Enumeration<Object>
public class StringTokenizerIteratorAdapter extends StringTokenizer implements Iterator<String> {public StringTokenizerIteratorAdapter(final String str, final String delim, final boolean returnDelims) {super(str, delim, returnDelims);}public StringTokenizerIteratorAdapter(final String str, final String delim) {super(str, delim);}public StringTokenizerIteratorAdapter(final String str) {super(str);}@Overridepublic boolean hasNext() {return hasMoreTokens();}@Overridepublic String next() {return nextToken();}
}
我们创建了一个双向「 two-way」适配器(相当于实现了 Enumeration 和 Iterator 两个接口),可以用作迭代器和StringTokenizer。迭代器方法不直接委托给枚举器中的方法,而是委托给StringTokenizer中更具体的方法。
相关文章:
适配器模式 Adapter Pattern
https://en.wikipedia.org/wiki/Adapter_pattern https://www.baeldung.com/java-adapter-pattern 适配器模式(也称为包装器「wrapper」,与装饰器模式「decorator pattern」共享的另一种命名),它允许将现有类的接口用作另一个接…...
Android 动态加入Activity 时 manifest 注册报错解决。使用manifestPlaceholders 占位
需求如下: 项目 测试demo 有多个渠道,部分渠道包含支付功能,在主测试代码外,需要一个单独 Activity 调用测试代码。 MainActivityPayActivity渠道A包含不包含渠道B包含包含 因为支付功能需要引入对应的 moudule,因此…...
芝加哥学派(Chicago School):金融与经济学的创新力量(中英双语)
芝加哥学派:金融与经济学的创新力量 在经济学和金融学的历史上,有一个学派的影响力不容忽视,那就是芝加哥学派(Chicago School)。芝加哥学派不仅在学术界广受推崇,也深刻影响了全球的经济政策和金融市场。…...
3分钟了解内外网文件传输:常见方法、注意事项有哪些?
内外网文件传输不仅是企业日常运营的基础设施,更是支持业务增长、创新和合规的关键工具。通过高效、安全的文件传输,企业能够更好地应对全球化协作、远程办公和数据安全等挑战,从而在竞争激烈的市场中保持领先地位。 一、内外网文件传输的常…...
Python学习心得常用的内置函数
常用的内置函数: 1.数据类型转换函数: 描述说明 描述说明 bool(obj) 获取指定对象 obj 的布尔值 str(obj) 将指定对象 obj 转成字符串类型 int(x) 将 x 转成 int 类型 float(x) 将 x 转成 float 类型 list(sequence) 将序列转成列表类型 tu…...
VMware Workstation16安装Centos7以及静态IP设置
配置虚拟机操作系统 1.创建新的虚拟机 -> 自定义配置,下一步 2. 选择虚拟机硬件兼容性 -> 默认,下一步 3.安装客户机操作系统 -> 稍后安装操作系统,下一步 4.选择客户机操作系统 -> LinuxCentOS7 64 位,下一步 5.命名…...
【核心算法篇十九】《 DeepSeek因果推断:双重差分模型如何破解政策评估的「时空难题」》
一、当AB实验不可行时,我们该相信什么?(因果推断困局解析) 假设某城市推出「夜间地铁免费」政策,市长想知道这个政策是否真的提升了夜间经济。这时候你会发现: 1️⃣ 无法克隆城市:不能同时存在一个「实施政策」和「不实施政策」的平行宇宙 2️⃣ 数据混杂严重:疫情反…...
Token Embedding(词嵌入)和Positional Encoding(位置编码)的矩阵形状关系及转换过程
在从零开始构建一个小型字符级语言模型时,简化的实现步骤是:数据准备→模型架构设计→训练→评估与生成。模型架构设计阶段的流程如下: 图1 模型架构设计阶段的流程 包含了输入层、嵌入层、解码器层和输出层。其中在嵌入层中包括了Token Embedding(词嵌入)和Positional En…...
多个用户如何共用一根网线传输数据
前置知识 一、电信号 网线(如以太网线)中传输的信号主要是 电信号,它携带着数字信息。这些信号用于在计算机和其他网络设备之间传输数据。下面是一些关于网线传输信号的详细信息: 1. 电信号传输 在以太网中,数据是…...
U-Net 与深度学习的完美结合:图像分割的高效解决方案
1. 引言:U-Net背景及应用 1.1 U-Net的起源与发展 U-Net 是由 Olaf Ronneberger 和他的团队于2015年提出的卷积神经网络(CNN)架构。最初的设计目的是解决医学图像分割中的挑战,尤其是在有限的训练数据下如何实现准确的分割。 在…...
nginx ngx_http_module(9) 指令详解
nginx ngx_http_module(9) 指令详解 nginx 模块目录 nginx 全指令目录 一、目录 1.1 模块简介 ngx_http_uwsgi_module:uWSGI支持模块,允许Nginx与uWSGI服务器进行通信。uWSGI是一种应用服务器协议,广泛用于Python Web应用的部署。通过该…...
【从0做项目】Java搜索引擎(4)——性能优化~烧脑~~~
本篇文章将对项目搜索引擎(1)~(3)进行性能优化,包括测试,优化思路,优化前后对比 目录 一:文件读取 二:实现多线程制作索引 1:代码分析 2:代码…...
【HarmonyOS Next】鸿蒙应用进程和线程详解
【HarmonyOS Next】鸿蒙应用进程和线程详解 一、前言 进程的定义: 进程是系统进行资源分配的基本单位,是操作系统结构的基础。 在鸿蒙系统中,一个应用下会有三类进程: (1) 主进程, (2) ExtensionAbility进程ÿ…...
【前端ES】ECMAScript 2023 (ES14) 引入了多个新特性,简单介绍几个不为人知但却好用的方法
Array.prototype.toSorted() 返回一个新的已排序数组副本,不改变原数组。 let arr [5, 4, 2, 3, 1]; console.log(arr.toSorted()); // [1, 2, 3, 4, 5]Array.prototype.with() 允许根据索引修改数组中的单个元素,并返回新数组。 const arr ["…...
【EndNote】WPS 导入EndNote 21
写在前面:有没有人有激活码,跪求! EndNote,在文献管理和文献引用方面很好用。写文章的时候,使用EndNote引入需要的文献会很方便。我目前用的WPS,想把EndNote的CWYW(Cite While You Write&#…...
网上购物|基于SprinBoot+vue的网上购物系统(源码+数据库+文档)
网上购物系统目录 基于SprinBootvue的网上购物 一、前言 二、系统设计 三、系统功能设计 5.1 管理员功能实现 5.1.1 论坛管理 5.1.2 商品管理 5.1.3 商品评价管理 5.1.4 商品订单管理 5.2 用户功能实现 5.2.1 商品信息 5.2.2 确认下单 5.2.3 商品订单 5.2.4 购物…...
AI 语言模型发展史:统计方法、RNN 与 Transformer 的技术演进
引言 自然语言处理(NLP)是 AI 领域的重要分支,而语言模型(Language Model, LM)是 NLP 的核心技术。语言模型经历了从 统计方法 到 RNN(循环神经网络),再到 Transformer 的演进&…...
Pycharm中查找与替换
1、Edit -> Find -> Find 在当前文件中查找 2、Edit -> Find -> Find in Files 在所有文件中查找 3、Edit -> Find -> Replace 在当前文件中执行替换 4、Edit -> Find -> Replace in Files 在所有文件中执行替换...
有向图的强连通分量: Kosaraju算法和Tarjan算法详解
在上一篇文章中, 我们了解了图的最小生成树算法. 本节我们来学习 图的强连通分量(Strongly Connected Component, SCC) 算法. 什么是强连通分量? 在 有向图 中, 若一组节点内的任意两个节点都能通过路径互相到达(例如 A → B A \rightarrow B A→B 且 B → A B \rightarro…...
mac相关命令
显示和隐藏usr等隐藏文件文件 terminal输入: defaults write com.apple.Finder AppleShowAllFiles YESdefaults write com.apple.Finder AppleShowAllFiles NO让.bashrc每次启动shell自动生效 编辑vim ~/.bash_profile 文件, 加上 if [ -f ~/.bashrc ]; then. ~/.bashrc fi注…...
如何掌握PostCSS fromJSON功能:AST序列化与反序列化的完整指南
如何掌握PostCSS fromJSON功能:AST序列化与反序列化的完整指南 【免费下载链接】postcss Transforming styles with JS plugins 项目地址: https://gitcode.com/gh_mirrors/po/postcss PostCSS是一个强大的CSS转换工具,它允许开发者使用JavaScrip…...
智慧农业害虫识别数据集 灯诱杀虫实验数据集 灯害虫数据集 常见农业害虫数据集 害虫手动标注数据集 24类常见农业害虫yolo格式 voc格式数据集地10172期
灯诱杀虫灯害虫数据集,常见农业害虫数据集。核心信息分类具体内容数据集名称灯诱杀虫灯害虫数据集、常见农业害虫数据集图像规模与划分共25378张jpeg图像;训练集12701张、验证集5077张、测试集7600张标注方式由农业高校相关教授手动标注适用任务害虫识别…...
全面掌握ezdxf:Python处理DXF文件的终极指南
全面掌握ezdxf:Python处理DXF文件的终极指南 【免费下载链接】ezdxf Python interface to DXF 项目地址: https://gitcode.com/gh_mirrors/ez/ezdxf 你是否曾经为处理复杂的CAD文件而感到头疼?🤔 特别是那些专有的DXF格式文件…...
SemScore:基于语义相似度的大语言模型评估方法
1. 项目概述SemScore是一种基于语义相似度的新型大语言模型(LLM)评估方法。在自然语言处理领域,传统的评估指标如BLEU、ROUGE等主要关注词汇层面的匹配程度,而SemScore则深入挖掘文本的语义内涵,通过计算生成文本与参考文本在语义空间中的相似…...
路由是什么
可以把网络世界想象成一张巨大的交通网,路由就是这张网里的导航系统。第一部分:基础概念——路由是什么,为什么需要它?1.1 没有路由,寸步难行你的电脑(192.168.1.5)想给公司服务器(1…...
设备预测性维护在物联网中的技术革新与应用实践
物联网技术的快速发展为工业设备管理带来了革命性变化,预测性维护(Predictive Maintenance, PdM)作为其核心应用之一,正在改变传统被动维修模式。通过实时数据采集、智能分析和故障预警,企业能够显著降低停机时间与维护…...
RAG系统在语义搜索中的架构设计与性能优化
1. 语义搜索中的RAG系统概述在信息检索领域,语义搜索技术正经历着从传统关键词匹配到深度理解用户意图的范式转变。RAG(Retrieval-Augmented Generation)系统作为这一转变中的关键技术框架,通过结合检索(Retrieval&…...
2026 全网最全内网渗透提权实战手册:Windows 与 Linux 双平台完整方法论
提权是内网渗透中最具决定性的环节,也是区分普通渗透测试人员与高级红队成员的核心能力。在2026年的攻防对抗中,传统的"打补丁就安全"的思维早已失效,攻击者正在利用越来越隐蔽的配置缺陷、系统特性滥用和新兴技术绕过防御体系。本…...
人类测试终局:智能化浪潮下的专业演进与价值重塑
在软件工程波澜壮阔的演进史中,测试始终扮演着沉默而关键的基石角色。它曾是质量防线的守门人,在代码与需求之间反复校验,确保交付物的可靠与稳定。然而,当大模型与智能体技术以前所未有的速度渗透至软件开发的全链路,…...
Depth-Anything-V2深度解析:单目深度估计基础模型的架构设计与实战应用
Depth-Anything-V2深度解析:单目深度估计基础模型的架构设计与实战应用 【免费下载链接】Depth-Anything-V2 [NeurIPS 2024] Depth Anything V2. A More Capable Foundation Model for Monocular Depth Estimation 项目地址: https://gitcode.com/gh_mirrors/de/D…...
