适配器模式详解

来源:互联网 发布:mysql substring函数 编辑:程序博客网 时间:2024/06/10 08:29

1  概述

        适配器模式把一个类的接口变换成客户端所期待的另一种接口,从而使原本接口不匹配而无法在一起工作的两个类能够在一起工作。


2  解决的问题

        使接口不兼容的类可以在一起工作。


3  模式中的角色

1)目标接口(Target):客户所期待的接口。目标可以是具体的或抽象的类,也可以是接口。

2)需要被适配的类(Adaptee):被适配的角色

3)适配器(Adapter):通过包装一个需要适配的对象,把原接口转换成目标接口。


4  模式分类

        类适配器模式和对象适配器模式。类适配器从已有的类继承;对象适配器容纳一个它包裹的类的实例。


5  类适配器模式


代码实现

// 已存在的、具有特殊功能、但不符合我们既有标准接口的类class Adaptee {    void SpecificRequst() {        System.out.println("...被适配类...具有特殊功能...");    }}// 目标接口(或标准接口)interface Target {    void Request();} // 具体目标类,只提供普通功能class ConcreteTarget implements Target {    public void Request() {        System.out.println("...普通类...具有普通功能...");    }}// 适配器类,继承了被适配类,同时实现标准接口class Adapter extends Adaptee implements Target {    public void Request() {        super.SpecificRequst();    }} // 测试类,即Clientpublic class ClassAdapterDemo {    public static void main(String[] args) {        // 使用普通功能类        Target concreteTarget = new ConcreteTarget();        concreteTarget.Request();                // 使用特殊功能类,即适配类        Target adapter = new Adapter();        adapter.Request();    }} 

运行结果:

...普通类...具有普通功能...

...被适配类…具有特殊功能...


        上面这种实现的适配器称为类适配器,因为 Adapter 类既继承了 Adaptee (被适配类),也实现了 Target 接口(因为 Java 不支持多继承,所以这样来实现),在 Client 类中我们可以根据需要选择并创建任一种符合需求的子类,来实现具体功能。


6  对象适配器模式


代码实现

// 已存在的、具有特殊功能、但不符合我们既有标准接口的类class Adaptee {    void SpecificRequst() {        System.out.println("...被适配类...具有特殊功能...");    }}// 目标接口(或标准接口)interface Target {    void Request();} // 具体目标类,只提供普通功能class ConcreteTarget implements Target {    public void Request() {        System.out.println("...普通类...具有普通功能...");    }}// 适配器类,直接关联被适配类,同时实现标准接口class Adapter implements Target {    // 直接关联被适配类    private Adaptee adaptee;        // 可以通过构造函数传入具体需要适配的被适配类对象    public Adapter(Adaptee adaptee) {        this.adaptee = adaptee;    }        public void Request() {        this.adaptee.SpecificRequst();    }} // 测试类,即Clientpublic class ObjectAdapterDemo {    public static void main(String[] args) {        // 使用普通功能类        Target concreteTarget = new ConcreteTarget();        concreteTarget.Request();                // 使用特殊功能类,即适配类        // 需要先创建一个被适配类的对象作为参数        Target adapter = new Adapter(new Adaptee());        adapter.Request();    }} 

运行结果:

...普通类...具有普通功能...

...被适配类…具有特殊功能...


        从类图和代码可以看出,Adapter 自身必须先拥有一个被适配类的对象,再把具体的特殊功能委托给这个对象来实现。使用对象适配器模式,可以使得 Adapter 类(适配类)根据传入的 Adaptee 对象达到适配多个不同被适配类的功能,当然,此时我们可以为多个被适配类提取出一个接口或抽象类。这样看起来的话,似乎对象适配器模式更加灵活一点。


7  应用举例

电源适配器

public class PowerAdapterDemo {    public static void main(String[] args) {        MobilePowerAdapter adpter = new MobilePowerAdapter();        int voltage = adpter.GetPower10V();        System.out.println("Voltage from adpater is " + voltage + "V");    }}interface ITarget {    int GetPower10V();}// 电源,被适配者(Adaptee)class Power {    int GetPower220V() {        return 220;    }}// 移动电源适配器class MobilePowerAdapter implements ITarget {    private Power power;    public MobilePowerAdapter() {        this.power = new Power();    }        @Override    public int GetPower10V() {        int v = power.GetPower220V();        System.out.println("Voltage from outlet is " + v + "V");                return change220To10(v);    }public int change220To10(final int voltage) {return voltage / 22;}} 

运行结果:

Voltage from outlet is 220V

Voltage from adpater is 10V


8  优点

1)通过适配器,客户端可以调用同一接口,因而对客户端来说是透明的。这样做的好处是更简单、更直接、更紧凑。

2)复用了现存的类,解决了现存类和复用环境要求不一致的问题。

3)将目标类和适配者类解耦,通过引入一个适配器类重用现有的适配者类,而无需修改原有的代码。


9  缺点

使系统结构更庞大,过多使用适配器会使系统结构非常凌乱,不容易进行整体把握。


10  适用场景

1)系统想使用现有的类,但这些类的接口不符合系统的接口

2)旧的系统已经实现了一些功能,但是客户端只能以另外接口的形式访问,但我们不希望手动更改原有类的时候

3)两个类所做的事情相同或相似,但具有不同接口的时候

4)想要建立一个可以重用的类,用于一些彼此之间没有大关联的类(包括一些可能在将来引进的类)一起工作

4)使用第三方组件,组件接口定义和自己定义的不同,不希望修改自己的接口,但是要使用第三方组件接口的功能

5)不适合在详细设计阶段使用,它是一种补偿模式,专用来在系统后期扩展、修改时使用


参考资料:

http://visionsky.blog.51cto.com/733317/384607

http://haolloyin.blog.51cto.com/1177454/346128

http://www.cnblogs.com/wangjq/archive/2012/07/09/2582485.html

0 0