设计模式之适配器模式

基本介绍

  • 适配器模式将某个类的接口转换成客户端期望的另外一个接口标识,主要目的是兼容性,让原本因接口不匹配而不能一起工作的两个类可以协同工作。其别名为包装器(wrapper)
  • 适配器模式属于结构型模式
  • 主要分为三类:类适配器模式、对象适配器模式、接口适配器模式

工作原理

  • 适配器模式:将一个类的接口转换成另一种接口,让原本不兼容的类可以兼容
  • 从用户的角度看不到被适配者,是解耦的
  • 用户调用适配器转化出来的目标接口方法,适配器再调用被适配者的相关接口方法
  • 用户收到反馈结果感觉只是和目标接口交互
    如下图:
    image

类适配器模式

基本介绍

adapter,通过继承src类,实现dst类接口,完成src->dst的适配

类适配器模式应用实例

  • 以充电器的例子来讲适配器,充电器本身相当于adapter,220V交流电相当于src,我们的目的dst是5V直流电
  • 思路分析图如下:
    image
  • 代码实现如下
public class Voltage220V {
    public int output220V(){
        int src = 220;
        System.out.println("output:"+src+"V");
        return src;
    }
}
public interface Voltage5V {
    int output5V();
}
public class VoltageAdapter extends Voltage220V implements Voltage5V {
    public int output5V() {
        int src = output220V();
        int dst = src / 44;
        return dst;
    }
}
public class Phone {
    public void charging(Voltage5V voltage5V) {
        if (voltage5V.output5V() == 5) {
            System.out.println("charging....");
        } else if (voltage5V.output5V() > 5) {
            System.out.println("bomb");
        }
    }
}
public class Client {
    public static void main(String[] args) {
        Phone phone = new Phone();
        phone.charging(new VoltageAdapter());
    }
}

注意事项和细节

  • java是单继承机制,所以类适配器需要继承src类这一点算是一个缺点,因为这要求dst必须是接口,有一定的局限性
  • src类的方法在Adapter中都会暴露出来,也增加了使用成本
  • 由于其继承了src类,所以它可以根据需求重写src类的方法,使得Adapter的灵活性增强了

对象适配器

对象适配器模式介绍

  • 基本思路和类的适配器模式相同,只是将Adapter类做修改,不是继承src,而是持有src类的实例,以解决兼容性的问题。即:持有src类,实现dst类接口,完成src->dst的适配
  • 根据“合成复用原则”,在系统中尽量使用关联关系来替代继承关系
  • 对象适配器模式是适配器模式常用的一种
    image
  • 代码如下:
public class Voltage220V {
    public int output220V(){
        int src = 220;
        System.out.println("output:"+src+"V");
        return src;
    }
}
public interface Voltage5V {
    int output5V();
}
public class VoltageAdapter implements Voltage5V {
    private Voltage220V voltage220V;

    public VoltageAdapter(Voltage220V voltage220V) {
        this.voltage220V = voltage220V;
    }

    public int output5V() {
        int dst = 0;
        if (voltage220V != null) {
            int src = voltage220V.output220V();
            dst = src / 44;
        }
        return dst;
    }
}
public class Phone {
    public void charging(Voltage5V voltage5V) {
        if (voltage5V.output5V() == 5) {
            System.out.println("charging....");
        } else if (voltage5V.output5V() > 5) {
            System.out.println("bomb");
        }
    }
}
public class Client {
    public static void main(String[] args) {
        Phone phone = new Phone();
        phone.charging(new VoltageAdapter(new Voltage220V()));
    }
}

对象适配器注意事项

  • 对象适配器和类适配器其实算是同一种思想,只不过实现方式不同。根据合成复用原则,使用组合代替继承,所以它解决了类适配器必须继承src的局限性问题,也不再要求dst必须是接口
  • 使用成本更低,更灵活

接口适配器模式(又名缺省适配器模式)

接口适配器模式介绍

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

适配器模式终极案例

image

public interface HandlerAdapter {

    /**
     * 判断是否支持这种Controller
     * @param controller
     * @return
     */
    Boolean support(Controller controller);

    /**
     * 执行该controller实际类型的调用方法
     * @param controller
     */
    void handle(Controller controller);
}
public class HttpHandlerAdapter implements HandlerAdapter {
    @Override
    public Boolean support(Controller controller) {
        if (controller instanceof HttpController) {
            return true;
        }
        return false;
    }

    @Override
    public void handle(Controller controller) {
        HttpController httpController = (HttpController) controller;
        httpController.doHandler();
    }
}
public class SimpleHandlerAdapter implements HandlerAdapter{
    @Override
    public Boolean support(Controller controller) {
        if (controller instanceof SimpleController) {
            return true;
        }
        return false;
    }

    @Override
    public void handle(Controller controller) {
        SimpleController httpController = (SimpleController) controller;
        httpController.doHandler();
    }
}
public class AnnotationHandlerAdapter implements HandlerAdapter {
    @Override
    public Boolean support(Controller controller) {
        if (controller instanceof AnnotationController) {
            return true;
        }
        return false;
    }

    @Override
    public void handle(Controller controller) {
        AnnotationController httpController = (AnnotationController) controller;
        httpController.doHandler();
    }
}
public interface Controller {
    /**
     * 实际执行业务的方法
     */
    void doHandler();
}
public class HttpController implements Controller {
    @Override
    public void doHandler() {
        System.out.println("do HttpController");
    }
}
public class SimpleController implements Controller{
    @Override
    public void doHandler() {
        System.out.println("do SimpleController");
    }
}
public class AnnotationController implements Controller {
    @Override
    public void doHandler() {
        System.out.println("do AnnotationController");
    }
}
public class DispatchServlet {
    public static Set<HandlerAdapter> handlerAdapters = new HashSet<HandlerAdapter>();

    public DispatchServlet() {
        handlerAdapters.add(new HttpHandlerAdapter());
        handlerAdapters.add(new SimpleHandlerAdapter());
        handlerAdapters.add(new AnnotationHandlerAdapter());
    }

    public void doDispatch(){
        /**
         * 这里的Controller具体类型可以从外部传进来,具体场景具体分析
         */
        SimpleController controller = new SimpleController();
        HandlerAdapter handlerAdapter = getHandler(controller);
        handlerAdapter.handle(controller);
    }

    public HandlerAdapter getHandler(Controller controller){
        for (HandlerAdapter handlerAdapter : handlerAdapters) {
            if (handlerAdapter.support(controller)) {
                return handlerAdapter;
            }
        }
        return null;
    }
}
public class Client {
    public static void main(String[] args) {
        DispatchServlet dispatchServlet = new DispatchServlet();
        dispatchServlet.doDispatch();
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值