介绍
适配器模式(Adapter Pattern)是作为两个不兼容的接口之间的桥梁。主要解决在软件系统中,常常要将一些”现存的对象”放到新的环境中,而新环境要求的接口是现对象不能满足的。适配器继承或依赖已有的对象,实现想要的目标接口。
适配器角色
- 目标(Target):定义一个客户端使用的特定接口。
- 客户端(Client):使用目标接口,与和目标接口一致的对象合作。
- 被适配者(Adaptee):一个现存需要适配的接口。
- 适配器(Adapter):负责将Adaptee的接口转换为Target的接口。
分类
适配器模式有两种:
1. 类适配器
所谓类适配器,指的是适配器Adapter继承我们的被适配者Adaptee,并实现目标接口Target。由于Java中是单继承,所以这个适配器仅仅只能服务于所继承的被适配者Adaptee。
2. 对象适配器
所谓对象适配器,简单的说就是适配器实现我们的目标接口,但是并不继承需要被适配的类。而是通过在适配器的构造函数中将需要被适配的类传递进来从而进行适配。
优缺点
共同点
优点:
- 将目标类和适配者类解耦
- 增加了类的透明性和复用性,将具体的实现封装在适配者类中,对于客户端类来说是透明的,而且提高了适配者的复用性
- 灵活性和扩展性都非常好,符合开闭原则
缺点:
- 过多地使用适配器,会让系统非常零乱,不易整体进行把握
- 由于 JAVA 至多继承一个类,所以至多只能适配一个适配者类,而且目标类必须是抽象类
不同点
类适配器优点:
由于适配器类是适配者类的子类,因此可以再适配器类中置换一些适配者的方法,使得适配器的灵活性更强。
类适配器缺点:
对于Java、C#等不支持多重继承的语言,一次最多只能适配一个适配者类,而且目标抽象类只能为接口,不能为类,其使用有一定的局限性,不能将一个适配者类和他的子类同时适配到目标接口。
对象适配器优点:
把多个不同的适配者适配到同一个目标,也就是说,同一个适配器可以把适配者类和他的子类都适配到目标接口。
对象适配器缺点:
与类适配器模式相比,要想置换适配者类的方法就不容易。
使用场景
有动机地修改一个正常运行的系统的接口
举例
定义被适配者(Adaptee)
public class Person {
/**
* person可以现在只能说英语
*/
public void sayEnglish(){
System.out.println("Person can say english!");
}
}
目标接口(Target)
public interface Target_Person {
void sayEnglish();
void sayFrench();
void sayChinese();
}
类适配器:
适配器(Adapter)
public class Adapter_Person extends Person implements Target_Person{
@Override
public void sayFrench() {
System.out.println("Person can say French!");
}
@Override
public void sayChinese() {
System.out.println("Person can say Chinese!");
}
}
客户端(Client)
public class Test {
public static void main(String[] args) {
Target_Person person = new Adapter_Person();
person.sayEnglish();
person.sayFrench();
person.sayChinese();
}
}
这个适配器Adapter只能为Person这一个类所服务。这时候如果需要适配的类很多,就要为每一个需要适配的类都写一个Adapter,有没有更加灵活的方式呢?那么就需要用到对象适配器。
对象适配器:
适配器(Adapter)
public class Adapter_Person implements Target_Person{ //仅仅实现目标接口
private Person person;
//在构造函数中将Adaptee类Person传递进来
public Adapter_Person(Person person){
this.person = person;
}
//实现目标接口中的sayEnglish()--调用Adaptee中的sayEnglish()
@Override
public void sayEnglish() {
this.person.sayEnglish();
}
//实现接口中的其它方法
@Override
public void sayFrench() {
System.out.println("person can say French!");
}
@Override
public void sayChinese() {
System.out.println("person can say Chinese!");
}
}
客户端(Client)
public class Test {
public static void main(String[] args) {
Target_Person person = new Adapter_Person(new Person());
person.sayEnglish();
person.sayFrench();
person.sayChinese();
}
}
补充:
android中的adapter用的是对象适配器