适配器模式

适配器模式简介

在现实生活中,经常出现两个对象因接口不兼容而不能在一起工作的实例,这时需要第三者进行适配。例如,讲中文的人同讲英文的人对话时需要一个翻译,用直流电的笔记本电脑接交流电源时需要一个电源适配器,用计算机访问照相机的 SD 内存卡时需要一个读卡器等。

在软件设计中也可能出现:需要开发的具有某种业务功能的组件在现有的组件库中已经存在,但它们与当前系统的接口规范不兼容,如果重新开发这些组件成本又很高,这时用适配器模式能很好地解决这些问题。

**适配器模式属于结构型模式 **

适配器模式涉及3个角色:

  • 源(Adaptee):需要被适配的对象或类型,相当于插头。
  • 适配器(Adapter):连接目标和源的中间对象,相当于插头转换器。
  • 目标(Target):期待得到的目标,相当于插座。

主要可以分为

  • 类适配器
  • 对象适配器
  • 接口适配器

适配器模式应用实例

以生活中充电器的例子来讲解适配器,充电器本身相当于Adapter,220V交流电 相当于src (即被适配者),我们的目dst(即 目标)是5V直流电

类适配器:

//被适配的类
public class Voltage220V {
	//输出220V的电压
	public int output220V() {
		int src = 220;
		System.out.println("电压=" + src + "伏");
		return src;
	}
}
//适配接口
public interface IVoltage5V {
	public int output5V();
}

//适配器类
public class VoltageAdapter extends Voltage220V implements IVoltage5V {

	@Override
	public int output5V() {
		// TODO Auto-generated method stub
		//获取到220V电压
		int srcV = output220V();
		int dstV = srcV / 44 ; //转成 5v
		return dstV;
	}

}
public class Phone {

	//充电
	public void charging(IVoltage5V iVoltage5V) {
		if(iVoltage5V.output5V() == 5) {
			System.out.println("电压为5V, 可以充电~~");
		} else if (iVoltage5V.output5V() > 5) {
			System.out.println("电压大于5V, 不能充电~~");
		}
	}
}

分析:我们的手机要使用5V的电压充电,就需要将220V的电压进行转换。

使用:

public class Client {

	public static void main(String[] args) {
		System.out.println(" === 类适配器模式 ====");
		Phone phone = new Phone();
		phone.charging(new VoltageAdapter());
	}

}

通过适配器类我们就解决了该问题。


类适配器模式注意事项和细节
Java是单继承机制,所以类适配器需要继承src类这一点算是一个缺点, 因为这要 求dst必须是接口,有一定局限性

src类的方法在Adapter中都会暴露出来,也增加了使用的成本。

由于其继承了src类,所以它可以根据需求重写src类的方法,使得Adapter的灵 活性增强了。

对象适配器

对象适配器模式介绍
基本思路和类的适配器模式相同,只是将Adapter类作修改,不是继承src类,而 是持有src类的实例,以解决兼容性的问题。 即:持有 src类,实现 dst 类接口, 完成src->dst的适配

根据“合成复用原则”,在系统中尽量使用关联关系来替代继承关系。

对象适配器模式是适配器模式常用的一种

以生活中充电器的例子来讲解适配器,充电器本身相当于Adapter,220V交流电 相当于src (即被适配者),我们的目dst(即目标)是5V直流电,使用对象适配器模 式完成。

只需要修改适配器类

//适配器类
public class VoltageAdapter  implements IVoltage5V {

	private Voltage220V voltage220V; // 关联关系-聚合
	//通过构造器,传入一个 Voltage220V 实例
	public VoltageAdapter(Voltage220V voltage220v) {
		this.voltage220V = voltage220v;
	}

	@Override
	public int output5V() {
		int dst = 0;
		if(null != voltage220V) {
			int src = voltage220V.output220V();//获取220V 电压
			System.out.println("使用对象适配器,进行适配~~");
			dst = src / 44;
			System.out.println("适配完成,输出的电压为=" + dst);
		}
		return dst;
	}
}

对象适配器和类适配器其实算是同一种思想,只不过实现方式不同。 根据合成复用原则,使用组合替代继承, 所以它解决了类适配器必须继承src的 局限性问题,也不再要求dst必须是接口。

接口适配器模式

一些书籍称为:适配器模式(Default Adapter Pattern)或缺省适配器模式。

  1. 当不需要全部实现接口提供的方法时,可先设计一个抽象类实现接口,并为该接 口中每个方法提供一个默认实现(空方法),那么该抽象类的子类可有选择地覆 盖父类的某些方法来实现需求
  2. 适用于一个接口不想使用其所有的方法的情况。
public interface Interface4 {
	public void m1();
	public void m2();
	public void m3();
	public void m4();
}

//在AbsAdapter 我们将 Interface4 的方法进行默认实现
public abstract class AbsAdapter implements Interface4 {
	//默认实现
	public void m1() {
	}
	public void m2() {
	}
	public void m3() {
	}
	public void m4() {
	}
}
public class Client {
	public static void main(String[] args) {
		AbsAdapter absAdapter = new AbsAdapter() {
			//只需要去覆盖我们 需要使用 接口方法
			@Override
			public void m1() {
				// TODO Auto-generated method stub
				System.out.println("使用了m1的方法");
			}
		};
	
		absAdapter.m1();
	}
}

适配器模式的注意事项和细节

适配器模式的注意事项和细节

  1. 三种命名方式,是根据 src是以怎样的形式给到Adapter(在Adapter里的形式)来 命名的。 2) 类适配器:以类给到,在Adapter里,就是将src当做类,继承 对象适配器:以对象给到,在Adapter里,将src作为一个对象,持有 接口适配器:以接口给到,在Adapter里,将src作为一个接口,实现
  2. Adapter模式最大的作用还是将原本不兼容的接口融合在一起工作。 4) 实际开发中,实现起来不拘泥于我们讲解的三种经典形式

源码举例

在开发中我们经常使用的HttpServlet,继承了GenericServlet,该类实现了Servlet接口,那为什么不直接使用HttpServlet实现Servlet接口呢??这里用的就是接口适配器模式,在GenericServlet中包含了很多空实现,这样HttpServlet在使用时,主需要覆写自己需要的方法即可

举例2:

JDK中的InputStreamReaderOutputStreamWriter类分继承了ReaderWriter接口,

但是要创建他们的对象必须在构造函数中传入一个InputStreamOutputStream的实例

InputStreamReaderOutputStreamWriter的作用就是将InputStreamOutputStream适配

ReaderWriter

如下图:

image-20201102234013112

此例中,InputStreamReader就是一个适配器类,

Reader就是我们的目标接口,

InputStream实例是我们的适配对象,

这里是通过构造器注入,属于对象适配器 , 这也是最常用的一种适配器模式

posted @ 2019-09-27 22:20  HeliusKing  阅读(255)  评论(0编辑  收藏  举报