浅学设计模式之适配器模式

    适配器模式(adapter pattern)有时候也称包装样式或者包装。将一个类的接口转接成用户所期待的。一个适配使得因接口不兼容而不能在一起工作的类工作在一起,做法是将类别自己的接口包裹在一个已存在的类中。

有两类适配器模式:

·对象适配器模式 - 在这种适配器模式中,适配器容纳一个它我包裹的类的实例。在这种情况下,适配器调用被包裹对象的物理实体。

·类适配器模式 - 这种适配器模式下,适配器继承自已实现的类(一般多重继承)。  

    下面我们可以看下类图:
浅学设计模式之适配器<Adapter>模式_第1张图片

    下面开始一个小例子。假如你朋友从日本给你带来一个日本原装相机,只能使用110V的电压充电,你在国内想使用的话必须使用一个电压转换器把国内的220V电压转换成110V电压。如下图

浅学设计模式之适配器<Adapter>模式_第2张图片


220V的电压转换成110V给客户。

      开始代码:

客户类:相机

public class Camera {
	public void charging(String strVoltage) {
		if (strVoltage.equalsIgnoreCase("110V")) {
			System.out.println("can charge");
		} else {
			System.out.println("the Voltage is not in conformity with the Camera");
		}
	}
}

这里只简单的实现了它的充电功能!

被适配者:220V中国电压

public class ChinaVoltage {
	 String getVoltage(){
		 System.out.println("中国电压是220V");
		 return "220V";
	 };
}

适配者:110V

public class JapanVoltage {
	private String strVoltage = "110V";
	public String getVoltage() {
		System.out.println("Japan Voltage is 110V");
		return strVoltage;
	}
}

这里没有定义接口,直接使用了实体类!

适配器:

public class VoltageAdapter extends ChinaVoltage{
	private JapanVoltage japanVoltage;
	
	public void setJapanVoltage(JapanVoltage japanVoltage) {
		this.japanVoltage = japanVoltage;
	}

	@Override
	public String getVoltage() {
		// TODO Auto-generated method stub
		return japanVoltage.getVoltage();
	}
	
}

测试用例:

public static void main(String[] args) {
		//得到相机
		Camera camera =new Camera();
		//得到日本电压和转换器
		JapanVoltage japanVoltage = new JapanVoltage();
		VoltageAdapter voltageAdapter = new VoltageAdapter();
		//转换成日本电压
		voltageAdapter.setJapanVoltage(japanVoltage);
		//充电
		camera.charging(voltageAdapter.getVoltage());
		
		//直接用中国电压充电
		camera.charging(new ChinaVoltage().getVoltage());
	}

测试结果:

Japan Voltage is 110V
can charge
中国电压是220V
the Voltage is not in conformity with the Camera


在这里使用的是对象适配器模式,在java中不支持多重继承,所以基本上没办法类继承方式。

但是使用接口加对象的方式试一下吧:

public interface IJapanVoltage {
	String getJapanVoltage();
}

适配器:


public class VoltageAdapter2 extends ChinaVoltage implements IJapanVoltage{

	@Override
	public String getJapanVoltage() {
		getVoltage();
		System.out.println("转换为日本电压为110V");
		return "110V";
	}

}



然后测试一下:

public class Test {
	public static void main(String[] args) {
		//得到相机
		Camera camera =new Camera();
		//得到日本电压和转换器
		JapanVoltage japanVoltage = new JapanVoltage();
		VoltageAdapter2 voltageAdapter = new VoltageAdapter2();
		//充电
		camera.charging(voltageAdapter.getJapanVoltage());
		
		//直接用中国电压充电
		camera.charging(voltageAdapter.getVoltage());
	}
}

测试结果:

中国电压是220V
转换为日本电压为110V
can charge
中国电压是220V
the Voltage is not in conformity with the Camera


适用性:

1. 系统需要使用现有的类,而此类的接口不符合系统的需要。

2. 想要建立一个可以重复使用的类,用于与一些彼此之间没有太大关联的一些类,包括一些可能在将来引进的类一起工作。这些源类不一定有很复杂的接口。

3. (对对象适配器而言)在设计里,需要改变多个已有子类的接口,如果使用类的适配器模式,就要针对每一个子类做一个适配器,而这不太实际。

 

效果及优缺点:

对于类适配器:

1. 用一个具体的Adapter类对Adaptee和Taget进行匹配。结果是当我们想要匹配一个类以及所有它的子类时,类Adapter将不能胜任工作。

2. 使得Adapter可以override(重定义) Adaptee的部分行为,因为Adapter是Adaptee的一个子类。

对于对象适配器:

1. 允许一个Adapter与多个Adaptee,即Adaptee本身以及它的所有子类(如果有子类的话)同时工作。Adapter也可以一次给所有的Adaptee添加功能。

2. 使得override(重定义)Adaptee的行为比较困难。如果一定要override Adaptee的方法,就只好先做一个Adaptee的子类以override Adaptee的方法,然后再把这个子类当作真正的Adaptee源进行适配。



最后:本文用图是从Head.First设计模式书上截的图。想学好设计模式,可以好好看看那本书。大笑

周末愉快。。。

如果觉得还可以的话,请帮忙顶一下吧。。。大笑



你可能感兴趣的:(设计模式,观察者模式,工作,Adapter,Adapter,测试)