适配器模式

定义

适配器模式 (Adapter) 将某个类的接口转换成客户期望的另一个接口表示,主要的目的是拥有兼容性,让原本因接口不匹配不能一起工作的两个类可以协同工作。

适配器模式属于类结构型模式。

主要分为两类:类(接口)适配器模式对象适配器模式
在这里插入图片描述
例如在欧洲国家使用美国制造的笔记本电脑,可能需要使用一个交流电插口适配器。

在这里插入图片描述

 

要点

在这里插入图片描述

  1. 从客户的角度看不到被适配者,是解耦的
  2. 客户调用适配器转化出来的目标接口方法,适配器再调用被适配者的相关接口方法
  3. 客户收到反馈结果,感觉只是和目标接口交互

类适配器使用继承方式,对象适配器使用组合方式。
 

场景

以生活中充电器的例子,充电器本身相当于 Adapter,220V 交流电相当于 src (即被适配者),我们的目的 dst (即目标)是 5V 直流电。

实现

1. 类适配器

IO220V

/**
 * 标压为220V的民用电(被适配的类)
 */
public class IO220V {

    public int output220V() {
        int src = 220;
        System.out.println("民用电电压为: " + src + "V.");
        return src;
    }

}

IO5V

/**
 * 输出5V标准电压(适配接口)
 */
public interface IO5V {

    int output5V();

}

Phone

/**
 * 手机
 */
public class Phone {

    public void change(IO5V io5V) {
        if (io5V.output5V() == 5) {
            System.out.println("手机正在充电...");
        } else {
            System.out.println("电压不达标,不能充电。");
        }
    }

}

PhoneAdapter

/**
 * 手机电源适配器(适配器类)
 */
public class PhoneAdapter extends IO220V implements IO5V {

    @Override
    public int output5V() {
        int dst = output220V() / 44;
        System.out.println("电源正在适配...");
        return dst;
    }

}

Client

public class Client {

    public static void main(String[] args) {
        Phone phone = new Phone();
        phone.change(new PhoneAdapter());
    }

}

注意: 类适配器往往需要多重继承,而Java 是单继承机制,所以类适配器必须要继承 IO220V 类,因为这要求 IO5V 必须是接口,有一定局限性。

当不需要全部实现接口提供的方法时,可先设计一个抽象类实现接口,并为该接口中每个方法提供一个默认实现(空方法),那么该抽象类的子类可有选择地覆盖父类的某些方法来实现需求,适用于一个接口不想使用其所有方法的情况。


2. 对象适配器
基本思路和类适配器相同,只是将 PhoneAdapter 类修改,不是继承 IO220V 类,而是持有 IO220V 类的实例,完成 IO220V -> IO5V 的适配。根据“合成复用原则”,在系统中尽量使用聚合(或组合)来替代继承。

IO220V

/**
 * 标压为220V的民用电(被适配的类)
 */
public class IO220V {

    public int output220V() {
        int src = 220;
        System.out.println("民用电电压为: " + src + "V.");
        return src;
    }

}

IO5V

/**
 * 输出5V标准电压(适配接口)
 */
public interface IO5V {

    int output5V();

}

Phone

/**
 * 手机
 */
public class Phone {

    public void charging(IO5V io5V) {
        if (io5V.output5V() == 5) {
            System.out.println("手机正在充电...");
        } else {
            System.out.println("电压不达标,不能充电。");
        }
    }

}

PhoneAdapter

/**
 * 手机电源适配器(适配器类)
 */
public class PhoneAdapter implements IO5V {

    private IO220V io220V;

    public PhoneAdapter(IO220V io220V) {
        this.io220V = io220V;
    }

    @Override
    public int output5V() {
        int dst = io220V.output220V() / 44;
        System.out.println("电源正在适配...");
        return dst;
    }
}

Client

public class Client {

    public static void main(String[] args) {
        Phone phone = new Phone();
        phone.charging(new PhoneAdapter(new IO220V()));
    }

}

注意:根据合成复用原则,使用组合替代继承, 所以它解决了类适配器必须继承 IO220V 的局限性问题,也不再要求 IO5V 必须是接口。

源代码

Click here

总结

对象适配器使用组合的方式,可以适配被适配类的任何子类,而类适配器使用继承的方式,在灵活性方面对象适配器更胜一筹。

在 SpringMVC 中的 HandlerAdapter, 就使用了适配器模式

在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值