Java学习记录--适配器模式

Java学习记录--适配器模式

标签(空格分隔): java


适配器模式是一种比较简单的设计模式,该博文从Java的Set集合入手,分析适配器模式适用场景,解决的问题,导致的缺点等.希望对你有帮助.


适配器模式,可以这样解释,用现有的组件去匹配实现要求提供的功能.举个例子:对于Java的HashSet集合,实际上是由一个HashMap映射来实现的,那么就可以理解为用现有的HashMap组件去实现了要求没有重复的Set集合这一功能.那么,这就是适配器模式,对于HashSet来说,更详细的叫法是对象适配器模式.


1.对象适配器模式

举个例子,MAC新款用的是雷电3接口,但是以前都是USB接口,那么就需要一个转换器,提供USB到雷电3的数据传输转换,那么在HashSet实现中每一个类都承担什么角色呢?

首先看HashSet的部分源码:

    //被适配的HashMap
    private transient HashMap map;
    //作为Map值的元素
    private static final Object PRESENT = new Object();
    //初始化被适配的HashMap
    public HashSet() {
        map = new HashMap<>();
    }

那么对于HashMap来说,HashMap就是USB接口,Set这个interface是新款的雷电3接口,于是HashSet就成了这个转换器,所提供的功能是把HashMap中的key单独表现为一个集合.

再看HashSet提供的转换:

public int size() {
        return map.size();
    }
public boolean contains(Object o) {
        return map.containsKey(o);
    }
public boolean add(E e) {
        return map.put(e, PRESENT)==null;
    }
public boolean remove(Object o) {
        return map.remove(o)==PRESENT;
    }

就像插口适配器提供数据传输转换一样,HashSet提供对HashMap映射的转换.这样来看的话很容易理解.参考下图:

Java学习记录--适配器模式_第1张图片
1.jpg

2.类适配器

如果把HashSet改造成下面这种形式的话,那就是类适配器,显然这种方式很不灵活,继承导致方法的混乱冲突,调用的不清晰等.因此建议使用对象适配器.

public class HashSet extends HashMap implements Set {

    private static final Object PRESENT = new Object();

    @Override
    public boolean add(K k) {
        return super.put(k, PRESENT) == null;
    }

    @Override
    public boolean remove(Object key) {
        return super.remove(key) == PRESENT;
    }
}

3.总结

3.1适用情景

  1. 系统需要使用现有的类,而这些类的接口不符合系统的接口。
  2. 想要建立一个可以重用的类,用于与一些彼此之间没有太大关联的一些类,包括一些可能在将来引进的类一起工作。
  3. 两个类所做的事情相同或相似,但是具有不同接口的时候。
  4. 旧的系统开发的类已经实现了一些功能,但是客户端却只能以另外接口的形式访问,但我们不希望手动更改原有类的时候。
  5. 使用第三方组件,组件接口定义和自己定义的不同,不希望修改自己的接口,但是要使用第三方组件接口的功能。

3.2 优点

  1. 通过适配器,客户端可以调用同一接口,因而对客户端来说是透明的。这样做更简单、更直接、更紧凑。
  2. 复用了现存的类,解决了现存类和复用环境要求不一致的问题。
  3. 将目标类和适配者类解耦,通过引入一个适配器类重用现有的适配者类,而无需修改原有代码.
  4. 一个对象适配器可以把多个不同的适配者类适配到同一个目标,也就是说,同一个适配器可以把适配者类和它的子类都适配到目标接口。

3.3缺点

  1. 对象适配器造成代码耦合,如果被适配的类,如HashMap由较为大的改动可能会影响适配器.另外更换适配器比较麻烦.
  2. 对于类适配器,不够灵活,写的不得当造成代码混乱.

根据自己的业务选择最适合的方式即可.

4.补充

4.1和装饰者模式比较

适配器像是转换器,使用新接口来调用原接口,并且有时候需要转换下功能,不应该对外直接暴露被适配的类的功能,而是通过自己本身的方法提供.如下面方法:

public boolean contains(Object o) {
        return map.containsKey(o);
    }

装饰者模式是提供新的职责,并且原封不动的提供被装饰者功能.

适配器是知道被适配者的详细情况的(就是那个类或那个接口)。装饰者只知道其接口是什么,至于其具体类型(是基类还是其他派生类)只有在运行期间才知道。

两者的有一个共同的名称叫做'包装模式',两者的最主要的区别在于使用目的不同,其他则大致思想一致.

参考博文:
http://blog.csdn.net/jason0539/article/details/22468457

你可能感兴趣的:(Java学习记录--适配器模式)