Java常用设计模式————适配器模式

引言

由于无法直接使用某个类中的方法而采取的一种中间类转换的策略。将一个类的接口转换成另一个接口,让原本接口不兼容的类可以兼容。

适配器模式可以分为三种:类适配器、对象适配器、接口适配器。它们之间的区别主要体现在适配器角色与被适配角色之间的依赖关系上。如类适配器是通过继承的方式,令适配器继承被适配类。

我们可以将适配器理解为两个不兼容的接口之间的桥梁。这是一种结构型模式。

虽然解决了老接口与新代码之间的兼容问题,但是适配器模式也存在不容忽视的缺点。过多的使用适配器会让系统变得凌乱,不易整体把握,因此,如果不是很有必要,可以不使用适配器,而是直接对系统进行重构。

注意,适配器不是在详细设计时添加的,而是解决正在服役的项目的兼容问题时添加的,而且不可以大量使用

一、类适配器

首先描述一下具体场景。

以电压适配为例。假设目前有一个用于输出电压的接口 Voltage,提供一个 output() 输出电压方法,且有一个可以输出220V标准家庭电压的实现类 Voltage220V。目前的系统中有一些家用电器的类,如TV(电视机类)、Fridge(电冰箱类)等,它们有一个charge() 方法,需要依赖 Voltage.output() 提供的电压为其充电。

Voltage220V的output()方法实现了 220V电压的输出功能,所以一切安好,但是今天我们买了一部手机,需要 5V的电压为其充电,但是目前系统中只有220V电压的输出,这时我们该如何解决呢?如下的类图所示:

此时我们最常想到的是再新建一个 Voltage 的实现类,可以叫 Voltage5V,然后实现其output() 方法。如下所示:

这种方法是一种解决思路,这并没有什么问题,实际上,在真正的生产开发中,也有很多是以这种方式解决的。当然,如果真的能够以这种方式解决问题是最好的。

但是,如果系统中 220V 电压是一个标准电压,我们无法新加一个与之等价的类,或者必须要基于已有的实现来解决的话,那么适配器模式就派上了用场,这也是适配器模式的一个主要特点,就是对已有方法的复用。于是,就有了下面这种适配器模式的类图:

代码实现如下:

/*** 电压接口*/
public interface Voltage {/** 电压输出*/int output();
}
/*** 220V电压实现*/
public class Voltage220V implements Voltage {private static final int voltage = 220;@Overridepublic int output() {System.out.println("输出" + voltage + "电压");return voltage;}
}
/*** 220V电压适配器*/
public class Voltage220VAdapter extends Voltage220V implements Voltage {@Overridepublic int output() {int superVoltage = super.output();System.out.println("父类输出电压:" + superVoltage + "V");int adaptedVoltage = superVoltage / 44;System.out.println("已将" + superVoltage + "V电压为" + adaptedVoltage + "V。");return adaptedVoltage;}
}
/*** 手机类*/
public class Phone {/*** 充电需要电压:5V*/public void charge(Voltage voltage) {if (voltage.output() == 5) {System.out.println("电压为5V,正在充电~");} else {System.out.println("电压异常,手机未充电!");}}
}

以上代码是适配器中几个关键角色,其中Voltage220V代表被适配类,Voltage220VAdapter 就代表220V电压的适配器类。以下是测试代码:

public class Test {public static void main(String[] args) {Phone phone = new Phone();phone.charge(new Voltage220VAdapter());}
}
// 输出:
输出220电压
父类输出电压:220V
已将220V电压为5V。
电压为5V,正在充电~

二、对象适配器

对象适配器与类适配器的区别是与被适配器类的关系,由泛化关系(继承)变为依赖关系(组合或聚合)。如下图所示:

可以看到,适配器类Voltage220VAdapter以组合的方式将Voltage220V的一个对象引用到其内部,这在一定程度上避免了因为继承关系带来的较强的耦合度。同时,该适配器类可以更加灵活,去适配更多的电压,如1000V等等。代码如下:

/*** 220V电压适配器*/
public class Voltage220VAdapter implements Voltage {private Voltage220V v220;public Voltage220VAdapter(Voltage220V v220) {this.v220 = v220;}@Overridepublic int output() {int srcVoltage = v220.output();System.out.println("原输出电压:" + srcVoltage + "V");int adaptedVoltage = srcVoltage / 44;System.out.println("已将" + srcVoltage + "V电压为" + adaptedVoltage + "V。");return adaptedVoltage;}
}

对象适配器与类适配器唯一的不同仅仅是对被适配类的依赖关系,由继承变为了组合或聚合,因此只有适配器本身稍有变化,去掉了继承关系,增加了一个 v220 字段。其他的接口都没什么变化。在测试代码中,我们需要为适配器类的构造器传入一个被适配的对象,如下所示:

public class Test {public static void main(String[] args) {Phone phone = new Phone();phone.charge(new Voltage220VAdapter(new Voltage220V()));}
}
// 输出:
输出220电压
原输出电压:220V
已将220V电压为5V。
电压为5V,正在充电~

三、接口适配器

接口适配器模式,也叫缺省适配器模式。

当不需要全部实现接口提供的方法时,可先设计一个抽象类实现接口,并为该接口中每个方法提供一个默认的空方法实现,这样,该抽象类的子类就可以有选择地覆盖父类的某些方法来实现需求。

接口适配器其实挺好理解的,在很多类库框架中也会用到,经常会看到一些方法仅仅是一个空方法,其实就用到这种接口适配器模式。相比于类适配器或对象适配器,这种接口适配器并不是为了实现某种兼容采用的解决办法,而更多的是为了保证各个子类的功能保持“单一职责原则”,避免将接口中不需要的方法暴露给外部调用者。

 

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

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

相关文章

Java常用设计模式————桥接模式

引言 在实际的业务中,经常会遇到多维度的概念组合,公园的门票,颐和园有年票、月票、日票,故宫也有年票、月票、日票。那么不同的公园和票种类型就可以视为两种不同的纬度,它们之间会形成相互组合的关系。 在类的设计…

Java常用设计模式————装饰者模式

引言 装饰者模式,又叫装饰器模式。它可以动态的将新功能附加到对象上。在对象功能扩展方面,它比继承更灵活,同时装饰者模式也体现了OCP原则。 在客户端调用使用了装饰者模式的对象时,就好像在使用构造器层层包裹核心对象&#x…

Java常用设计模式————组合模式

引言 组合模式,是一种类似递归算法的结构性设计模式,通过以简单的 List ,组合本类对象,实现树状对象结构的“部分、整体”的层次。 它可以让调用程序不需要关心复杂对象与简单对象的区别,而统一地实现处理逻辑。 对…

Java常用设计模式————外观模式

引言 外观模式(Facade Pattern),又叫“过程模式”。外观模式为子系统中的一组接口提供一个一致的入口,此模式定义了一个高层接口,这个接口使得这一组子系统更加易用。 一、案例分析 生活中有很多类似的案例&#xf…

Java常用设计模式————享元模式

引言 享元模式,也叫蝇量模式(Flyweight Pattern)。运用共享技术有效地支持大量细粒度的对象。 享元模式常用于系统底层开发,解决系统的性能问题。例如数据库连接池,里面都是创建好的连接对象,在这些连接对…

IDEA——常用基础设置

一、设置入口 File—>Settings... 或者 在工具栏的“小扳手”图标。 二、主题设置 三、编辑通用设置 设置面板中的 Editor 3.1 自动导包 可以设置IDEA自动为程序导包,在书写时加入准确的导包,在书写时优化导包(自动去掉未使用的&#…

IDEA——常用快捷键

引言 总结 IDEA 的常用快捷键,除了部分快捷键与 Eclipse 保持一致之外,枚举更多的实用快捷键。 一、如何设置快捷键 在 Settings -> Keymap 中,下拉框里选择 Eclipse ,即可将 IDEA 的快捷键设置为与 Eclipse 保持一致。但并…

IDEA——常用代码模板

引言 IDEA 提供了一些内置的代码模板,可以让开发者快速方便的使用,当然 eclipse 中也是有的,比如输入 syso 快速生成输出语句,main 快速生成主函数等。 idea 的模板设置都在 Settings --> Live Templates 和 General-->Po…

IDEA——Git 的设置与使用

引言 在本机下载好 Git 之后,再去在 IDEA 中设置 Git 相关的参数。详细的 Git 操作和 Eclipse 大同小异,可以移步至:《Git必知必会》 一、设置Git执行程序路径 二、导入一个新的远程 git 托管项目 打开 File ——> New ——> Project…

IDEA——Maven的配置与使用

引言 简单介绍一下如何在 idea 中配置maven,以及如何去使用 maven 。 一、配置 Maven home Maven home 和 settings 文件一般都需要进行重新设置,关联到本机已经安装好的 maven 版本,settings 这里可以使用默认,也可以设置为 ma…

Spring Cloud Alibaba——Nacos实现服务治理

引言 本博客总结微服务开发中各个微服务调用的实现,并使用 Nacos 完成服务注册和发现。 文章中会涉及到 maven 的使用,以及 spring boot 的一些知识。开发工具采用 IDEA 2020.2。 设计一个电商订单和商品购买微服务,实现微服务的注册发现与…

Spring Cloud —— Feign 实现服务调用

引言 本篇博客简单介绍 Feign 的基础知识和基本应用,以前一篇博客《Spring Cloud Alibaba——Nacos实现服务治理》为代码基础,实现更简单的微服务调用方式。 一、什么是Feign restTemplate 实现的微服务调用方式: // 调用商品微服务&…

Spring Cloud —— 负载均衡与 Ribbon 应用

引言 本篇博客简单介绍微服务负载均衡的概念,并通过 IDEA 多端口启动应用的方式,模拟多个应用实例,使用自定义和 Ribbon 两种方式实现基本的负载均衡策略。 微服务代码以《Spring Cloud Alibaba——Nacos实现服务治理》为基础。 一、什么是…

Spring —— 容器内部逻辑

引言 上一篇关于IoC容器的详解《Spring —— IoC 容器详解》真是工程浩大,可以说Spring官网对核心中的核心IOC容器做了非常全面的使用说明,包括在《Spring揭秘》中让我一直没有成功的Method Injection,官网也解决了我的疑惑,并最…

2020 年度总结

2020年给我的感觉是短平快的一年。 由于年初的新冠肺炎疫情,我大半年都呆在北京的破旧出租屋里写代码。整个春天和夏天,平平无奇。 2月1日返京,居家办公,夜跑。8月复工,疯狂爆痘、烂脸,月末落户天津。9月…

JDBC——概述与JDBC的使用

引言 一直希望深入学习一下数据库持久化技术,接触过Hibernate、Mybatis,也使用过Spring事务管理来控制回滚操作,但是越发觉得底层知识有一定的知识盲区和空洞。 很多ORM框架都是基于JDBC规范来进行构建的,因此,学习J…

JDBC——编程式事务的实现逻辑

引言 数据库事务的概念和基础,总结在《MySQL 基础 ————事务与隔离级别总结》。 本篇博客通过“JDBC 纯编码”方式实现事务控制,完成一个 A 给 B 转账的小功能,在进一步熟练JDBC的编程流程的同时,重点关注 Java 语言如何操作…

排序算法——插入排序

一、算法思想 给定一个无序数列,模拟一个指针从第0位开始向后,始终保持当前位置左边的数列是有序的。 指针位置上的元素依次与前面的元素比较,当遇到小于自己的数或右边已经没有元素时,都停止比较,开始下一轮。 生活…

Linux 底层原理 —— epoll 与多路复用

引言 epoll 是 Linux 系统下高性能网络服务的必备技术,很多面试中高频出现的 Nginx、Redis 都使用了这一技术,本文总结 linux 多路复用模型的演变过程,看一看epoll 是如何实现高性能的。 一、相关基础知识 1.1 文件描述符 文件描述符&…

异或运算的应用

一、基础知识 异或运算,相异为1。 异或运算是一种常用的位运算,在算法题中,对于避免额外的空间复杂度有独特的用处。 异或运算也被称为“无进位相加”,它具有以下特性: 特性1:0 ^ N N 特性2&#xff1a…