设计模式 | 适配器模式(adapter)

定义:

将一个类的接口转换成客户希望的另外一个接口。Adapter模式使得原本由于接口不兼容而不能一起工作的那些类可以一起工作。
 
书中说到Gof的设计模式中,讲了两种类型的适配器模式:
1.类适配器模式
2.对象适配器模式
其中,类适配器模式,是通过多重继承来实现了。
但是Java是不支持多重继承的,所以下面主要讲的是对象适配器模式。
 

结构:(书中图,侵删)

 

一个客户真正要使用的目标接口(具体一点说可以叫方法),包含它的可以是接口、抽象类、类
一个需要被适配的接口(方法),同上
一个适配器,继承自目标接口,包含一个需要被适配的接口
 

实例:

书中提到一个电源适配器的例子,很形象,所以这里提一嘴。
就是中国的电源是220V,但有的国家是110V,出国之后依旧想使用中国的电器怎么办?就需要用到适配器来转换一下电压。
说到这里,我就想到了一个例子,护照。
中国公民有中国的身份证,外国人也有外国的id card,我们不能指望外国人都用中国的身份证,同样我们也不可能都使用外国的id card。
但是出入境怎么验证你的身份呢,这就需要用到护照,护照就相当于全球通用的身份证。我们这里忽略签证的存在,签证也要盖在护照上不是。
 
目标类(外国身份认证):
package designpattern.adapter;

import java.util.HashSet;
import java.util.Set;

public class ForeignIdentify {
    String socailSecurityNumber;

    // 模拟一个社保号的数据库
    static Set<String> dbSet = new HashSet<>();
    static {
        dbSet.add("F001");
        dbSet.add("F002");
        dbSet.add("F003");
    }

    public boolean check(String socailSecurityNumber) {
        if (dbSet.contains(socailSecurityNumber)) {
            System.out.println(socailSecurityNumber + "是外国的合法公民");
            return true;
        } else {
            System.out.println(socailSecurityNumber + "不是外国的合法公民");
            return false;
        }

    }
}
被适配类(中国身份认证):
package designpattern.adapter;

import java.util.HashSet;
import java.util.Set;

public class ChineseIdentify {
    String idCardNumber;

    // 模拟一个身份证号的数据库
    static Set<String> dbSet = new HashSet<>();
    static {
        dbSet.add("张三C001");
        dbSet.add("李四C002");
        dbSet.add("王五C003");
    }

    public boolean check(String idCardNumber, String name) {
        if (dbSet.contains(name + idCardNumber)) {
            System.out.println(name + idCardNumber + "是中国的合法公民");
            return true;
        } else {
            System.out.println(name + idCardNumber + "不是中国的合法公民");
            return false;
        }

    }
}
适配器(护照):
package designpattern.adapter;

import java.util.HashMap;
import java.util.Map;

public class ChinesePassport extends ForeignIdentify {
    ChineseIdentify chineseIdentify = new ChineseIdentify();
    
    // 模拟身份证、名字对应数据库
    static Map<String, String> db = new HashMap<>();
    static {
        db.put("C001", "张三");
        db.put("C002", "李四");
        db.put("C003", "王五");

    }

    @Override
    public boolean check(String idCardNumber) {
        // 为了模拟两个接口不完全一样,假设中国的身份验证需要身份证号和名字两个条件
        return chineseIdentify.check(idCardNumber, getName(idCardNumber));
    }

    private String getName(String idCardNumber) {
        return db.get(idCardNumber);
    }
}
客户端:
package designpattern.adapter;

public class Client {
    public static void main(String[] args) {

        ForeignIdentify foreignIdentify = new ForeignIdentify();
        enter("F001", foreignIdentify);
        enter("F004", foreignIdentify);

        // 直接用身份证号来验证
        enter("C003", foreignIdentify);
        // 换成护照
        foreignIdentify = new ChinesePassport();
        enter("C003", foreignIdentify);
    }

    public static void enter(String number, ForeignIdentify foreignIdentify) {
        System.out.println("=============外国入关==============");
        if (foreignIdentify.check(number)) {
            System.out.println("允许入关!");
        } else {
            System.out.println("禁止入关!");
        }

    }
}

结果输出:

=============外国入关==============
F001是外国的合法公民
允许入关!
=============外国入关==============
F004不是外国的合法公民
禁止入关!
=============外国入关==============
C003不是外国的合法公民
禁止入关!
=============外国入关==============
王五C003是中国的合法公民
允许入关!

总结:

这个模式准确的说是一个无奈的选择,不应该当做第一选择,当需要适配的两样东西不好改动的时候,为了让他们能匹配上,才不得不使用适配器。
如果能够提前避免接口不匹配的情况,及时进行代码的重构是更好的选择。就像如果说有电子设备都是同样的充电口,那我们就不需要准备他们多条充电线了。
posted @ 2019-06-03 20:05  莫愆  阅读(781)  评论(0编辑  收藏  举报