java 动态代理

来源:互联网 发布:六小龄童知乎扒皮 编辑:程序博客网 时间:2024/06/05 20:37

         java代理可以分为三类:静态代理、JDK动态代理以及CGLIB动态代理。

       其中,像包装器模式就属于静态代理,具体参考相应的设计模式。另外,直接继承目标类,也可以实现静态代理。静态代理方式,代理类和目标对象耦合度比较高。

       java自带的动态代理就是JDK动态代理。主要通过Proxy代理类以及调用处理接口InvocationHandler来实现。下面重点说明下这种代理方式。

        >.1. Interface InvocationHandler:

           该接口是代理实例的调用处理程序需要实现的接口。每个代理实例都具有一个关联的调用处理程序。对代理实例调用方法时, 将对方法调用进行编码并将其指派到它的调用处理程序的invoke 方法。

           该接口下只包含一个方法, Object  invoke(Object proxy,Method method,Object[] args)

           其中,proxy---代理实例;

                      method---代理实例执行目标对象的方法实例;

                      args---执行方法的参数数组。

         >.2. Proxy 即代理类:

            提供了用于创建动态代理类和实例的静态方法。

                 主要方法:Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)

             该方法返回一个指定接口的代理类实例,

             其中,loader---定义代理类的类加载器;

                        interfaces---代理类实现的接口列表;

                        h---指派方法调用的调用处理程序。

             其他具体方法可以参考JDK文档。

            下面通过一个具体的Demo来演示动态代理的实现:

             首先,定义一个接口

       public interface ISubject       {    public void doWork();       }

               然后,定义一个该接口的具体实现类

       public class ASubjectImpl implements ISubject       {   @Override   public void doWork()   {System.out.println("A Subject do something!");   }       }

               接着,是一个代理实现类

       public class ProxyHandler implements InvocationHandler       {   private Object proxied;   public ProxyHandler(Object proxied)   {this.proxied = proxied;   }   @Override   public Object invoke(Object proxy, Method method, Object[] args)throws Throwable   {// 在调用对象方法之前可以做一些工作//...//执行代理对象方法Object result = method.invoke(proxied, args);//执行代理对象方法之后执行一些操作return result;   }       }

               最后,测试代码:

        public static void main(String args[]){//创建代理对象ISubject proxySubject = (ISubject) Proxy.newProxyInstance(ISubject.class.getClassLoader(), new Class[]{ISubject.class}, new ProxyHandler(new ASubjectImpl()));//执行代理操作proxySubject.doWork();}

              执行结果很明显。上面代码的执行过程如下所示:

           

               对目标对象的调用,被Proxy拦截,并在InvocationHandler的回调方法中通过反射调用目标对象。

           Java自带的动态代理的限制在于只能对实现了接口的目标类进行代理,即接口代理。要想对类进行代理,可以通过CGLIB来实现。CGLIB采用生成目标类的子类的技术创建代理对象,所以不能对目标类的finally方法或者finally类进行代理。

           另外,CGLIB底层直接采用ASM编码方式,因此创建的代理对象性能更高。但是,创建代理对象的过程还是JDK动态代理性能更好。因此,对于单例对象来说,只创建一次,可以采用CGLIB动态代理,而非单例对象,因为每次都是创建新的对象,可以考虑代用JDK动态代理。

          Spring AOP的底层实现就是通过动态代理来支持的。下节会分析下Spring AOP的动态代理实现。



0 0
原创粉丝点击