Java动态代理

代理是一种常用的程序设计模式,如同网络代理一样,代理是介于调用者和真正调用目标对象之间的中间对象,代理在调用真正目标对象时提供一些额外或者不同的操作,真正的对目标对象的操作还是通过代理调用目标对象来完成。

    简单的代理例子如下:

[java]  view plain copy
  1. //接口  
  2. interface Interface{  
  3.     void doSomething();  
  4.     void somethingElse(String arg);  
  5. }  
  6. //目标对象  
  7. class RealObject implement Interface{  
  8.     public void doSomething(){  
  9.         System.out.println(“RealObject doSomething”);  
  10. }  
  11. public void somethingElse(String arg){  
  12.     System.out.println(“RealObject somethingElse ” + arg);  
  13. }  
  14. }  
  15. //简单代理对象  
  16. class SimpleProxy implements Interface(  
  17.     private Interface proxied;  
  18.     public SimpleProxy(Interface proxied){  
  19.         this.proxied = proxied;  
  20. }  
  21. public void doSomething(){  
  22.     System.out.println(“SimpleProxy doSomething”);  
  23.     proxied.doSomething();  
  24. }  
  25. public void somethingElse(String arg){  
  26.     System.out.println(“SimpleProxy somethingElse ” + arg);  
  27.     proxied.somethingElse(arg);  
  28. }  
  29. )  
  30. Class SimpleProxyDemo{  
  31.     public static void consumer(Interface iface){  
  32.         iface.doSomething();  
  33.         iface.somethingElse(“TestProxy”);  
  34. }  
  35. public static void main(String[] args){  
  36. //不是用代理  
  37.     cosumer(new RealObject());  
  38.     //使用代理  
  39.     cosumer(new SimpleProxy(new RealObject()));  
  40. }  
  41. }  

输出结果为:

RealObject doSomething

RealObjectsomethingElse TestProxy

SimpleProxy doSomething

RealObject doSomething

SimpleProxy somethingElse TestProxy

RealObject somethingElse TestProxy

上面例子可以看出代理SimpleProxy在调用目标对象目标方法之前做了一些额外的操作。

Java中的代理是针对接口的动态代理,当然java也可以使用第三方的CGLIB实现针对类的代理,但是JDK中只支持针对接口的动态代理,我们只分析JDK的动态代理。

JDK动态代理的要素:

(1).实现了InvocationHandler的代理处理类,实现其invoke方法,该方法是代理调用目标对象方法以及提供额外操作的方法。

(2).使用Proxy.newProxyInstance(类加载器, 代理接口列表,InvocationHandler对象);方法创建实现了指定接口的动态代理。

JDK的代理例子如下:

[java]  view plain copy
  1. //接口  
  2. interface Interface{  
  3.     void doSomething();  
  4.     void somethingElse(String arg);  
  5. }  
  6. //目标对象  
  7. class RealObject implement Interface{  
  8.     public void doSomething(){  
  9.         System.out.println(“RealObject doSomething”);  
  10. }  
  11. public void somethingElse(String arg){  
  12.     System.out.println(“RealObject somethingElse ” + arg);  
  13. }  
  14. }  
  15. //代理处理类  
  16. class DynamicProxyHandler implements InvocationHandler{  
  17.     provate Object proxied;  
  18.     public DynamicProxyHandler(Object proxied){  
  19.         this.proxied = proxied;  
  20. }  
  21. //动态代理调用目标对象的方法  
  22. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable{  
  23.     System.out.println(“Dynamic proxy invoke”);  
  24.     return method.invoke(proxied, args);  
  25. }  
  26. }  
  27. class SimpleDynamicProxy{  
  28.     public static void consumer(Interface iface){  
  29.         iface.doSomething();  
  30.         iface.somethingElse(“DynamicProxy”);  
  31. }  
  32. public static void main(String[] args){  
  33.     RealObject real = new RealObject();  
  34.     //不是用代理  
  35.     consumer(real);  
  36.     //创建动态代理  
  37.     Interface proxy = (Interface) Proxy.newProxyInstance(  
  38.                     Interface.class.getClassLoader(),  
  39.                     new Class[]{Interface.class},  
  40.                     new DynamicProxyHandler(real)  
  41. );  
  42.        cosumer(proxy);  
  43. }   
  44. }  

输出结果为:

RealObject doSomething

RealObject somethingElse DynamicProxy

Dynamic proxy invoke

RealObject doSomething

Dynamic proxyinvoke       

RealObject somethingElse DynamicProxy




InvocationHandler 用于实现代理。 

如果不用InvocationHandler接口实现代理的话,我们写代码是这样的: 
定义一个接口:

Java code
    
    
interface Greet { void sayHello(String name); void goodBye(); }



实现这个接口:

Java code
    
    
class GreetImpl implements Greet { public void sayHello(String name) { System.out.println( " Hello " + name); } public void goodBye() { System.out.println( " Good bye. " ); } }



实现一个代理类

Java code
    
    
public class SimpleProxy implements Greet { private Greet greet = null ; SimpleProxy(Greet greet) { this .greet = greet; } public void sayHello(String name) { System.out.println( " --before method sayHello " ); greet.sayHello(name); System.out.println( " --after method sayHello " ); } public void goodBye() { System.out.println( " --before method goodBye " ); greet.goodBye(); System.out.println( " --after method goodBye " ); } /** * @param args */ public static void main(String[] args) { Greet greet = new SimpleProxy( new GreetImpl()); // 生成代理 greet.sayHello( " walter " ); greet.goodBye(); } }




代理其实没什么的,再看看如果实现了InvocationHandler接口, 
我们怎样实现代理。 
还是要实现原来的Greet接口。 
接口的实现还是GreetImpl。

Java code

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

public class DebugProxy implements java.lang.reflect.InvocationHandler
{
    private Object obj;

    public static Object newInstance(Object obj)
    {
        return java.lang.reflect.Proxy.newProxyInstance(obj.getClass().getClassLoader(),
                obj.getClass().getInterfaces(), new DebugProxy(obj));
    }

    private DebugProxy(Object obj)
    {
        //Greet接口的實現:GreetImpl
        this.obj = obj;
    }

    //Method m:調用的方法
    //Object[] args:方法要傳入的參數
    public Object invoke(Object proxy, Method m, Object[] args) throws Throwable
    {
        Object result;
        try
        {
            //自定義的處理
            System.out.println("--before method " + m.getName());
            //調用GreetImpl中方法
            result = m.invoke(obj, args);
        }
        catch(InvocationTargetException e)
        {
            throw e.getTargetException();
        }
        catch(Exception e)
        {
            throw new RuntimeException("unexpected invocation exception: " + e.getMessage());
        }
        finally
        {
            System.out.println("--after method " + m.getName());
        }
        return result;
    }

    /**
     * @param args
     */
    public static void main(String[] args)
    {
        Greet tmp = new GreetImpl();
        
        Greet greet = (Greet) DebugProxy.newInstance(tmp);
        //生成的greet和tmp有相同的hashCode
        
        greet.sayHello("walter");
        greet.goodBye();
    }
}

实现了InvocationHandler接口的

 Object invoke(Object proxy, Method method, Object[] args)

Proxy 代理类

Method 调用的方法

Args  调用方法的参数

的方法的类可以作为InvocationHandler类的参数来构建Proxy类的实例


当调用被代理的类(Connection)的实例的方法时系统将转到包含代理类(Proxy)的InvocationHandlerinvoke方法中去执行相应代码。

这样就可以在不修改代码的前提下为已经存在的类来添加新的功能。

如: 

  public Object invoke(Object proxy, Method m, Object[] args) throws Throwable {

    Object obj = null;

    if (CLOSE_METHOD_NAME.equals(m.getName())) {

      DBConnectionPool.pushConnectionBackToPool(this);

    }

    else {

      obj = m.invoke(m_originConnection, args);

    }

    if (DEBUG) {

        System.out.println(m.getName()+"is invoke!");

    }

    lastAccessTime = System.currentTimeMillis();

    return obj;

  }



评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值