读书人

设计方式感触之代理模式应用

发布时间: 2012-08-13 13:21:53 作者: rapoo

设计模式感触之代理模式应用

?

设计模式感触之代理模式应用

?

如果说看完设计模式之后,哪个模式最让我印象深刻和半醉半醒,那一定就是代理模式(Proxy)。代理模式看似非常简单,很直接,应用的也很广泛,然而,放下书,去使用的时候,可能是由于动态代理和远程代理实现的细节还没弄清,忽然发现,代理模式究竟为哪般还是模糊的。此处记录和总结下我所理解的代理模式,本部分主要是基础部分。

?

public interface Callable {    public void call();}
??

?

public class Phone implements Callable {    @Override    public void call() {        System.out.println("Phone is calling ...");    }    //Other Phone methods}
?

这时候,出现了一个新的需求,就是需要对Phone的所有动作进行计时,以对最终的收费提供依据,这时候有三种解决方案:

?

public class Phone implements Callable {    @Override    public void call() {        long startTime = System.currentTimeMillis();        System.out.println("Phone is calling ...");        long endTime = System.currentTimeMillis();        System.out.println("Call duration = " + (endTime - startTime));    }    //Other Phone methods}
?

?

public class Phone2 extends Phone {    @Override    public void call() {        long startTime = System.currentTimeMillis();        super.call();        long endTime = System.currentTimeMillis();        System.out.println("Call duration = " + (endTime - startTime));    }}
?

?

public class Phone3 implements Callable {    private Callable phone;     public Phone3(Callable phone) {        this.phone = phone;    }     @Override    public void call() {        long startTime = System.currentTimeMillis();        phone.call();        long endTime = System.currentTimeMillis();        System.out.println("Call duration = " + (endTime - startTime));    }}
?

仔细观察,聚合为什么会具有更强的能力呢,这里主要是在Phone3里面传入的是Callable接口,因此非常灵活,其实这是另外一个面向对象设计原则:依赖反转原则:要依赖于抽象,不要依赖与具体。另外一种说法是:要针对接口编程,不要针对实现编程。

代理,按其字面意义来说,就是指代替其他人做他们该做的事情,广泛的来讲,上述的继承和聚合方式实现都是代理的形式,继承中,子类将call的行为交给父类处理,聚合中,将具体的call行为交给引用的类处理。

在GoF提出的设计模式中,代理模式是这样定义的:

对其他对象提供一种代理以控制对这个对象的访问。

由此可见,代理与真实对象实现了同一个接口,对真实对象的请求事实上是通过代理来处理的。也可以说明前面所述的继承与聚合事实上也是一种代理。

上面所述的都属于静态代理,但代理模式的一大犀利之处就是其也能实现动态代理,即在运行时动态决定真实的被代理对象。

Java自身提供了实现动态代理的结构,自定义动态代理类,只需要使用java.lang.reflect包下面的InvocationHandler, Proxy和Method三个类就可以了,下面看一个简单的例子:

?

public class FirstInvocationHandler implements InvocationHandler {    private Object o;     public FirstInvocationHandler(Object phone) {        this.o = phone;    }     @Override    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {        System.out.println("Invoke start!");        method.invoke(o, args);        System.out.println("Invoke completed!");        return null;    }}
?

?

public class TestJavaProxy {     public static void main(String[] args) {        Phone phone = new Phone();        Callable proxyPhone = (Callable) Proxy.newProxyInstance(phone.getClass().getClassLoader(), phone.getClass().getInterfaces(), new FirstInvocationHandler(phone));        proxyPhone.call();               Date date = new Date();        Serializable proxyDate = (Serializable)Proxy.newProxyInstance(date.getClass().getClassLoader(), date.getClass().getInterfaces(), new FirstInvocationHandler(date));        proxyDate.toString();    }}
?

通过例子,将可以发现,我们可以自由实现我们需要的代理,只需要能传递正确的classLoader、interfaces和args,就能被代理到,这对于具有通用增强功能方法的尤其有意义,比如在典型的日志记录、事务管理、权限等控制上非常灵活。

本节主要论述代理模式中的一些应用,对JDK的动态代理也做了一些例子,关于更深层次的代理感触请查看下一节。

读书人网 >软件开发

热点推荐