Spring源码分析spring源码Java学习笔记

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

2017-08-30  本文已影响200人  阿亮私语

前言

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

1、jdkproxy

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

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

参数说明:

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()方法。
操作步骤如下:

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();
    }
}

输出结果如下:

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

总结:

区别

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

jdkProxy

优点:

cglib

优点:

上一篇 下一篇

猜你喜欢

热点阅读