适配器模式

结构型模式

结构型模式概述

结构型模式(Structural Pattern)描述如何将类或者对象结合在一起形成更大的结构,就像搭积木,可以通过简单积木的组合形成复杂的、功能更为强大的结构。

结构型模式可以分为类结构型模式和对象结构型模式:

  • 类结构型模式关心类的组合,由多个类可以组合成一个更大的系统,在类结构型模式中一般只存在继承关系和实现关系
  • 对象结构型模式关心类与对象的组合,通过关联关系使得在一个类中定义另一个类的实例对象,然后通过该对象调用其方法。
  • 根据“合成复用原则”,在系统中尽量使用关联关系来替代继承关系,因此大部分结构型模式都是对象结构型模式。

适配器模式

模式定义

适配器模式(Adapter Pattern) :将一个接口转换成客户希望的另一个接口,适配器模式使接口不兼容的那些类可以一起工作,其别名为包装器(Wrapper)。适配器模式既可以作为类结构型模式,也可以作为对象结构型模式。

模式结构

  • Target(目标抽象类):目标抽象类定义客户所需接口,可以是一个抽象类或接口,也可以是具体类。
  • Adapter(适配器类):适配器可以调用另一个接口,作为一个转换器,对Adaptee和Target进行适配,适配器类是适配器模式的核心,在对象适配器中,它通过继承Target并关联一个Adaptee对象使二者产生联系。
  • Adaptee(适配者类):适配者即被适配的角色,它定义了一个已经存在的接口,这个接口需要适配,适配者类一般是一个具体类,包含了客户希望使用的业务方法,在某些情况下可能没有适配者类的源代码。

类适配器

类适配器

1
2
3
4
5
public class Adapter extends Adaptee implements Target{
public void request(){
specificRequest();
}
}

对象适配器

对象适配器

1
2
3
4
5
6
7
8
9
public class Adapter extends Target{
private Adaptee adaptee;
public Adapter(Adaptee adaptee){
this.adaptee=adaptee;
}
public void request(){
adaptee.specificRequest();
}
}

模式实例

某公司打算开发一个根据邮编代号来判断地点的功能,目前开发方已开发了一个类 C ,类中有方法 string findzipcode(int zipcode[5]),在输入美国邮编的情况下得到美国的地址。假定现在又想开发中国邮编查询的功能,这时开发方获得了一个中国邮编查询的类 U,该类也有类似的方法 string zipcode(int code[6]),请用适配器模式来解决这个问题。(注:中国邮编是6位,美国邮编是5位)。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
//目标抽象类
public interface Target {
public void findzipcode(String zipcode);
}

public class C implements Target{
public void findzipcode(String zipcode){
System.out.println("邮编:"+zipcode+";美国地址:xxxx");
}
}

public class U {
public void findAddress(String zipcode){
System.out.println("邮编:"+zipcode+";中国地址:xxxx");
}
}

public class UAdapter extends U implements Target {
@Override
public void findzipcode(String zipcode) {
super.findAddress(zipcode);
}
}

public class TestOfMail {
public static void main(String[] args) {
Target a1 = new UAdapter();
a1.findzipcode("545454");

Target a2 = new C();
a2.findzipcode("54545");
}
}

模式优缺点

适配器模式的优点

  • 将目标类和适配者类解耦,通过引入一个适配器类来重用现有的适配者类,而无须修改原有代码。
  • 增加了类的透明性和复用性,将具体的实现封装在适配者类中,对于客户端类来说是透明的,而且提高了适配者的复用性。
  • 灵活性和扩展性都非常好,通过使用配置文件,可以很方便地更换适配器,也可以在不修改原有代码的基础上增加新的适配器类,完全符合“开闭原则”

类适配器模式还具有如下优点:

  • 由于适配器类是适配者类的子类,因此可以在适配器类中置换一些适配者的方法,使得适配器的灵活性更强。

类适配器模式的缺点如下:

  • 对于Java、C#等不支持多重继承的语言,一次最多只能适配一个适配者类,而且目标抽象类只能为抽象类,不能为具体类,其使用有一定的局限性,不能将一个适配者类和它的子类都适配到目标接口。

对象适配器模式还具有如下优点:

  • 一个对象适配器可以把多个不同的适配者适配到同一个目标,也就是说,同一个适配器可以把适配者类和它的子类都适配到目标接口。

对象适配器模式的缺点如下:

  • 与类适配器模式相比,要想置换适配者类的方法就不容易。如果一定要置换掉适配者类的一个或多个方法,就只好先做一个适配者类的子类,将适配者类的方法置换掉,然后再把适配者类的子类当做真正的适配者进行适配,实现过程较为复杂。

模式适用环境

在以下情况下可以使用适配器模式:
系统需要使用现有的类,而这些类的接口不符合系统的需要。
想要建立一个可以重复使用的类,用于与一些彼此之间没有太大关联的一些类,包括一些可能在将来引进的类一起工作。

文章目录
  1. 1. 结构型模式
    1. 1.1. 结构型模式概述
  2. 2. 适配器模式
    1. 2.1. 模式定义
    2. 2.2. 模式结构
      1. 2.2.1. 类适配器
      2. 2.2.2. 对象适配器
    3. 2.3. 模式实例
    4. 2.4. 模式优缺点
      1. 2.4.1. 适配器模式的优点
    5. 2.5. 模式适用环境

20160528-DesignPattern-6/

本页二维码