进阶之路:Java设计模式---适配器模式

el/2024/4/19 23:42:46
适配器模式定义:

适配器模式就是把一个类的接口变为另一个类期望的接口,从而让原来两个因接口不匹配而无法一起工作的两个类能在一起工作

适配器模式:

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

类适配器模式实现:
/*** 目标类*/
public interface Target {void operationOne();void operationTwo();}
/*** 源类* 内部包含operationOne但不包含operationTwo*/
public class Adaptee {public void operationOne(){}}
/*** 扩展了Adaptee又实现了Target*/
public class Adapter extends Adaptee implements Target {/*** 该函数由Adaptee继承而来*/@Overridepublic void operationOne() {super.operationOne();}/*** 该函数由Target实现而来*/@Overridepublic void operationTwo() {}
}

适配器角色Adapter实现了源类Adaptee,同时又实现了Target,由于Adaptee没有operationTwo()函数,但是目标接口又要求这个接口,因此适配器角色Adapter实现了这个函数。

对象适配器模式实现:
/*** 目标类*/
public interface Target {void operationOne();void operationTwo();}
/*** 源类* 内部包含operationOne但不包含operationTwo*/
public class Adaptee {public void operationOne(){}}
public class Adapter{private Adaptee adaptee;public Adapter(Adaptee adaptee) {this.adaptee = adaptee;}/*** 源类Adaptee含有operationOne函数,适配器类直接委派即可*/public void operationOne(){adaptee.operationOne();}/*** 源类Adaptee不含有operationTwo函数,适配器类需要补充该函数*/public void operationTwo(){}
}
类适配和对象适配模式的优缺点:
  • 类适配器使用对象的继承方式,是静态的定义方式;对象适配器使用对象组合的方式,是动态组合方式
  • 对于类适配器模式,由于适配器直接继承了Adaptee,使得适配器不能和Adaptee的子类一起工作,因为继承是静态的关系,当适配器继承了Adaptee后,就不可能再去处理Adaptee的子类了
  • 对于对象适配器模式,一个适配器可以把多种不同的源适配到同一个目标。也就是说,同一个适配器可以把源类和它的子类都适配到目标接口。因为对象适配器采用的是对象组合的关系,只要对象类型正确,是不是子类都无所谓
  • 对于类适配器模式,适配器可以重新定义Adaptee的部分行为,即子类重写父类的函数
  • 对于对象适配器模式,重新定义Adaptee的行为比较困难,这种情况下,需要定义Adaptee的子类来实现定义,然后让适配器组合子类。虽然重定义Adaptee的行为比较困难,当时想要增加新的行为比较方便,而且新增的行为适用于所有的源
  • 对于类适配器模式,仅仅引入了一个对象,并不需要额外的引用来简介得到Adaptee
  • 对于对象适配器,需要额外的引用来间接得到Adaptee建议尽量使用对象适配器的实现方式,多用合成/聚合、少用继承。当然,具体问题具体分析,根据需要来选用实现方式,最适合的才是最好的。
适配器模式优缺点:

优点:

  • 良好的复用性:系统需要使用现有的类,但是现有类接口不符合系统需求,通过适配器模式可以实现更好的复用
  • 良好的扩展性:在实现适配器功能的时候,可以调用自己开发的功能,从而自然的扩展系统的功能

缺点:

  • 过多的适配器会让系统零乱,不易整体把握。比如,明明看到调用的是A接口,其实内部被适配成了B接口的实现,一个系统如果太多出现这种情况,无异于一场灾难。因此如果不是很有必要,可以不使用适配器,而是直接对系统进行重构。

http://www.ngui.cc/el/5181697.html

相关文章

一天或者几天一个小算法---数列排序方式

/*** 数字排序*/ public class Algorithm5 {/*** 冒泡排序* 自认为是最简单的一个排序方法,只是拿出一个数,大的放后,小的放前就可以了* 时间复杂度为O(n2)*/public static int [] sortOfBubble(int [] valueArray){int temp;for (int i 0 ;…

进阶之路:Java设计模式---外观模式

外观模式定义: 外部与子系统间进行通信时,需要通过一个统一的外观进行,为子系统的一系列接口提供一个统一的界面。外观模式定义了一个高层接口,这个接口使子系统更易使用。外观模式又称为门面模式,它是一种对象型结构…

Android崩溃日志收集---CrashHandler,给你自残的理由

使用CrashHandler的目的 测试人员过来告诉你,“你写的代码崩溃了!” “what ? ! 我写的代码怎么可能会崩溃?!你复现一下,污蔑我的话信不信自残给你看” “哎,这次怎么好了,什么情况?” 虽然说这种几率性bug可以晚点解决,但是终归是要解决的,但是复现又有点困难,那…

关于线程安全最通俗易懂的解释

下面的博客是在博客园上面看到的大佬写的,写的很好,通俗易懂,作为一个萌新看完以后促使我对线程安全有了新的理解。 博客未经博主的同意,所以未擅自转载,下面附快速通道 点次进入博客内容

Android 获取目录下所有文件信息(名字、大小)

private void getFiles(){File file = new File(GlobalConstant.POSLOGPATH);File[] files = file.listFiles();for (int i = 0; i < files.length; i++) {File childFile = files[i]

Java开发人员最常犯的9个错误

1.Array转ArrayList 当需要把Array转成ArrayList的时候&#xff0c;开发人员经常这样做&#xff1a; String [] strArray {"0" , "1" , "2"}; List<String> list Arrays.asList(intArray);Arrays.asList会返回一个ArrayList&#xff0…

Fragment的show()、hide()和attach()、detach()切换方式生命周期过程

1.使用show()、hide()方式进行fragment切换 @Overridepublic void onClick(View view) {switch (view.getId()){case R.id.main_home:switchFragmentWithShowHide(0);break;case R.id.main_shop:switchFragmentWithShowHide(1);break;case R.id.main_community:switchFragmentW…

aar包导入引起的Error:Failed to resolve: :xxxaar: 的问题

在Android Studio中 如果在module中依赖了aar库&#xff0c;当build工程的时候&#xff0c;会出现failed to resolve的错误 想要解决这个问题可以在app的build.gradle中加入下面代码&#xff1a; repositories {flatDir {dirs libs}}举个栗子&#xff1a;在app的build.gradle里…

Android Studio新建module,以及module引用aar等相关问题

一、新建module如果是作为lib被引用&#xff0c;需要选择Android Library。 二、作为lib的module如果引用了aar文件&#xff1a; 1、把aar文件放到module的libs目录下 2、作为lib的module的gradle文件&#xff1a; repositories {flatDir {dirs libs} }dependencies {.....…

Mysql在Spring中配置hibernate的字符编码格式

真正的配置非常简单&#xff0c;只需要在链接字符串加上对应的参数即可&#xff0c;如下示例&#xff1a;jdbc:mysql://localhost:3306/zbdx_jxdd?useUnicodetrue&characterEncodingUTF-8 useUnicodetrue &characterEncodingUTF-8是编码相关的配置&#xff0c;useUni…