适配器模式 一个简单场景
程序员文章站
2022-05-05 12:00:57
...
最近在重构旧的代码,感觉代码异常的糟糕,连配置文件都是在java代码中写死的。
这个不是本文的重点,本文主要通过一个例子来慢慢分析适配器的模式,对适配器模式可以先看下
http://haolloyin.blog.51cto.com/1177454/346128这个文章,写的不错,但是感觉还是有点抽象。
我们这里以java编程思想(第4版)的15章的泛型中提到的适配来讲解。(部分代码做了修改,并加了点自己的理解)
首先有一个Fill的类。
public class Fill { //初始化size个classToken放到collection里 public static <T> void fill(Collection<T> collection, Class<? extends T> classToken, int size) { for(int i = 0; i < size; i++){ collection.add(classToken.newInstance()); } } }
再有一个比较简单的队列。
public class SimpleQueue<T> implements Iterable<T> { private LinkedList<T> storage = new LinkedList<T>();//LinkedList 也是一个simple队列 public void add(T t) {storage.offer(t);} public T get() {return storage.poll();} public Iterator<T> iterator(){ return storage.iterator(); } }
写个测试用例:
public class FillTest{ public static void main(String[] args){ List<String> list1 = new ArrayList<>(); Fill.fill(list1,String.class,2); SimpleQueue<String> queue1 = new SimpleQueue<>(); //下面的代码将会编译失败 Fill.fill(queue1,String.class,2); } }
问题出来了,Fill类竟然不能处理SimpleQueue,我这里面也有add方法啊。是哪出问题了,对了,就是这个Fill类有问题,由于Fill设计的不合理,导致无法处理除Collection以外的类了。
那我们重新定义Fill类,这里命名为Fill2
//定义一个可add的接口 interface Addable<T> {void add(T t);} public class Fill2{ public static <T> void fill(Addable<T> addable,Class<? extends T> classToken, int size){ for(int i= 0;i < size ; i++){ try{ addadle.add(clasToken.newInstance()); }catch(Exception e){ throw new RuntimeException(e); } } } }
这里我们定义了Fill2,它可以支持所有实现Addable接口的类,但是Collection没有实现Addable接口啊,那怎么办?就需要一个适配类了。
class AddableCollectionAdapter<T> implements Addable<T>{ private Collection<T> c; //一个参数为Collection的构造器 public AddableCollectionAdapter(Collection<T> c){ this.c = c; } public void add(T item){ c.add(item); } }
这里由于Collection类是java内置的接口,我们不易使用类适配器(两个接口都有add()方法,Colleciont还有大量别的方法),最好使用对象适配器。 同理我们也可以为Simplequeue添加一个适配类,但Simplequeue是自己定义,使用类适配器会更好点:
class AddableSipleQueue<T> extends SimpleQueue<T> Addable<T> { public void add(T item ){super.add(item);} }
下面我们测试下:
class Fill2Test { public static void main(String[] args){ List<String> list = new ArrayList<>; Fill2.fill(new AddableColletionAdapter(list),String.class,2); Addable<String> queue = new AddableSimpleQueue<>(); Fill2.fill(queue,String.class,2); } }
写到这里再抛出个问题,如果还是用Fill类,那能不能写SimpleQueue适配类来适应Fill里的Collection?
总结下:适配有2种方式:类适配器和对象适配器。
适配器一般用在前期设计不合理导致的,需要后期的修改,一般少用。