设计模式(八):适配器模式

一、概述

  适配器模式将一个类的接口,转换为客户期望的另一个接口。适配器让原本不兼容的类可以合作无间

二、解决问题

  从模式的定义中,我们看到适配器模式就是用来转换接口,解决不兼容问题的。想想我们现实生活中的适配器,最常用的就是手机充电器了,也叫做电源适配器,它把家用交流强电转换为手机用的直流弱电。其中交流电就是被适配者,充电器是适配器,手机是用电客户。

三、结构类图

  

四、成员角色

  客户(Client):只能调用目标接口功能,不能直接使用被适配器,但可以通过适配器的接口转换间接使用被适配器。

  目标接口(Target):客户看到的接口,适配器必须实现该接口才能被客户使用。

  适配器(Adapter):适配器把被适配者接口转换为目标接口,提供给客户使用。

  被适配者(Adaptee):被适配者接口与目标接口不兼容,需要适配器转换成目标接口子类,才能被客户使用。

五、应用实例

  下面用鸟叫和鸟飞的例子解析适配器模式,鹦鹉会叫也会飞,但鹅就只会叫不会飞,而且鹅也不是鸟类,我们要创建一个适配器,把鹅转换成鸟。

  第一步、创建鸟接口,对应角色目标接口

package adapter.pattern;

//鸟接口
public interface Bird {
	//鸟叫
	public void chirp();
	//飞
	public void fly();
}

  第二步、创建鹦鹉类

package adapter.pattern;

//鹦鹉类实现鸟接口
public class Parrot implements Bird{

	public void chirp() {
		System.out.println("呜呜呜");		
	}

	public void fly() {
		System.out.println("我能飞很远很远");		
	}

}

  第三步、创建鹅类,对应角色被适配者

package adapter.pattern;

//鹅类,不是鸟
public class Goose {
	//鹅会叫但不会飞,没有飞的方法
	public void chirp(){
		System.out.println("嘎嘎嘎");
	}
}

  第四步、创建适配器

package adapter.pattern;

//适配器,把鹅类适配为鸟
public class GooseAdapter implements Bird{
	//组合鹅类
	Goose goose;
	public GooseAdapter(Goose goose){
		this.goose = goose;
	}
	
	public void chirp() {
		if(goose != null){
			//调用鸟叫的时候委托鹅叫
			goose.chirp();
		}
	}

	public void fly() {
		//不支持该操作,可以抛出该异常,客户可以知道详情
		throw new UnsupportedOperationException();
	}

}

  第五步、测试适配器

package adapter.pattern;

public class AdapterTest {
	public static void main(String[] args){
		System.out.println("-----鹦鹉会叫也会飞-----");
		Bird parrot = new Parrot();
		parrot.chirp();
		parrot.fly();
		
		System.out.println("-----鹅会叫但不会飞-----");
		Goose goose = new Goose();
		Bird gooseAdapter = new GooseAdapter(goose);
		gooseAdapter.chirp();
		gooseAdapter.fly();
		
		
	}
}

  运行结果:

  适配器在java中的例子,我们知道ArrayList类实现了迭代器(Iterator),但不支持枚举(Enumeration),下面我们就来实现ArrayList的枚举操作。

package adapter.pattern;

import java.util.ArrayList;
import java.util.Enumeration;

public class ItertorEnumeration implements Enumeration{
	//组合被适配者
	private ArrayList<String> list;
	//集合的计数器,判断指针指向集合的位置
	private int index = 0;
	public ItertorEnumeration(ArrayList<String> list){
		this.list = list;
	}
	public boolean hasMoreElements() {
		if(list != null && list.size() > index){
			return true;
		}
		return false;
	}

	public Object nextElement() {
		String content = list.get(index);
		index++;
		return content;
	}
}

  

六、优点和缺点

  1、优点

  (1)、转换接口,适配器让不兼容的接口变成兼容。

  (2)、让客户和实现的接口解耦。有了适配器,客户端每次调用不兼容的接口时,不用修改自己的代码,只要调用适合的适配器就可以了。

  (3)、使用了对象组合设计原则。以组合的方式包装被适配者,被适配者的任何子类都可以搭配着同一个适配器使用。

  (4)、体现了“开闭”原则。适配器模式把客户和接口绑定起来,而不是和具体实现绑定,我们可以使用多个配适器来转换多个后台类,也可以很容易地增加新的适配器。

  2、缺点

  (1)、每个被适配者都需要一个适配器,当适配器过多时会增加系统复杂度,降低运行时的性能。

  (2)、实现一个适配器可能需要下一番功夫,增加开发的难度。

七、使用场景

  1、当要使用的两个类所做的事情相同或者相似,但是具有不同的接口时考虑使用配适器模式。

  2、当需要统一客户端调用接口的代码,而所调用的接口具有不兼容问题时使用适配器模式。这样客户端只有调用一个接口就行了,这样可以更简单、更直接、更紧凑。

八、总结

  1、适配器有对象适配器和类适配器,类适配器需要用到多重继承。

  2、适配器就是转换接口达到我们的需要。

posted @ 2016-08-07 15:49  jenkinschan  阅读(1983)  评论(2编辑  收藏  举报