美文网首页Spring源码分析spring源码Java学习笔记
Spring源码-AOP(二)-jdkProxy与cglib

Spring源码-AOP(二)-jdkProxy与cglib

作者: 阿亮私语 | 来源:发表于2017-08-30 16:23 被阅读200次

    前言

    上一篇分享了一些AOP相关的概念,这一篇继续上一篇分享java动态代理的两种实现方式。

    1、jdkproxy

    缺陷:JDK的动态代理依靠接口实现,如果类没有实现接口,则不能使用jdk代理,只能使用cglib,但是这也比静态代理好太多。
    jdkproxy中包含一个类和一个接口

    InvocationHandler接口: 
    public interface InvocationHandler { 
    public Object invoke(Object proxy,Method method,Object[] args) throws Throwable; 
    } 
    

    参数说明:

    • Object proxy:指被代理的对象。
    • Method method:要调用的方法
    • Object[] args:方法调用时所需要的参数
      代码示例
      先定义接口
    public interface BookFacade {
        void addBook();
    }
    

    定义该接口的具体实现类

    public class BookFacadeImpl implements BookFacade{
        @Override
        public void addBook() {
            System.out.println("--------->【添加图书】<---------");
        }
    }
    
    

    实现jdk代理接口InvocationHandler

    /**
     * created by sunliangliang
     * 代理类,jdk代理必须事先InvocationHandler接口
     */
    public class BookFacadeProxy implements InvocationHandler{
        private Object target;
    
        public Object bind(Object target){
            this.target = target;
            System.out.println("-----------bind------");
           return Proxy.newProxyInstance(target.getClass().getClassLoader(),target.getClass().getInterfaces(),this);//this即绑定当前
        }
        @Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            Object result = null;
            System.out.println("----------->【任务执行前】<-----------");
            result = method.invoke(target,args);
            System.out.println("----------->【任务结束】<-----------");
            return result;
        }
    }
    

    测试类

    public class TestProxy {
        public static void main(String[] args) {
            BookFacadeProxy proxy = new BookFacadeProxy();
            BookFacade bookProxy = (BookFacade) proxy.bind(new BookFacadeImpl());
            bookProxy.addBook();
        }
    }
    
    

    那我们看下输出结果

    -----------bind------
    ----------->【任务执行前】<-----------
    --------->【添加图书】<---------
    ----------->【任务结束】<-----------
    
    

    总结:由此可以看出使用JDK代理必须实现InvocationHandler接口,
    将具体业务类绑定到Proxy这个类上,然后会自动执行invoke()方法。
    操作步骤如下:

    • 1、接口和业务实现类编写
    • 2、代理类实现InvocationHandler接口,将实现类绑定到JDK代理类Proxy上,如下
      • Proxy.newProxyInstance(target.getClass().getClassLoader(),target.getClass().getInterfaces(),this);//this即绑定当前代理类,target是指实现类
    • 3、调用测试类实现,调用绑定方法会即可执行,具体有invoke()执行

    2、cglib

    在用cglib的过程中遇到了一个巨坑,希望大家重点关注jar包。我此处引入的jar只有一个
    cglib-nodep-3.2.5.jar,该jar包中已经包含asm.jar的包,若是cglib-2.x.jar的版本需要引入asm.jar,因为asm.jar的版本发生过变更,所以会出现版本冲突的问题。切记jar环境。
    接口和实现类用的同上,代理类代码如下:

    public class BookFacadeCglibProxy implements MethodInterceptor{
        private Object target;
    
        /**
         * 创建代理对象
         * @param target
         * @return
         */
        public Object getInstance(Object target){
            this.target = target;
            Enhancer enhancer = new Enhancer();
            enhancer.setSuperclass(target.getClass());
            // 回调方法
            enhancer.setCallback(this);
            //创建代理对象
            return enhancer.create();
        }
        // 回调方法
        @Override
        public Object intercept(Object obj, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
            System.out.println("---------->【开始执行任务】<--------");
            methodProxy.invokeSuper(obj,args);
            System.out.println("---------->【结束执行任务】<--------");
            return null;
        }
    }
    
    

    测试类如下:

    public class TestCglib {
        public static void main(String[] args) {
            BookFacadeCglibProxy cglib=new BookFacadeCglibProxy();
            BookFacadeImpl bookFacade = (BookFacadeImpl) cglib.getInstance(new BookFacadeImpl());
            bookFacade.addBook();
        }
    }
    

    输出结果如下:

    ---------->【开始执行任务】<--------
    --------->【添加图书】<---------
    ---------->【结束执行任务】<--------
    

    总结:

    • 1、代理实现MethodInterceptor接口,

    区别

    两者最大的区别是jdkproxy需要统一的接口,而cglib不需要。

    jdkProxy

    优点:

    • 不依赖第三方jar包, 使用方便
    • 随着jdk升级,性能稳定提升
      缺点:
    • 只能代理实现接口的类
    • 执行速度慢
      适用场景:
      如果你的程序需要频繁、反复地创建代理对象,则JDK动态代理在性能上更占优。

    cglib

    优点:

    • 由于是动态生成字节码实现代理,因此代理对象的执行速度较快, 约为JDK动态代理的1.5 ~ 2倍
    • 可以代理没有实现接口的对象
      缺点:
    • 不能代理final类
    • 动态生成字节码虽然执行较快,但是生成速度很慢
      适用场景:
      不需要频繁创建代理对象的应用,如spring中默认的单例bean,只需要在容器启动时生成一次代理对象。

    相关文章

      网友评论

        本文标题:Spring源码-AOP(二)-jdkProxy与cglib

        本文链接:https://www.haomeiwen.com/subject/gnfxjxtx.html