Java及Spring框架中的AOP技术汇总--使用CGLib对接口和类进行动态代理及限制条件

上一篇中我们使用JDK 内置的功能来进行接口方法的拦截(AOP)处理。但是限制是必须要使用接口方式。

本篇换成使用CGLib【Code Generation Library】这个库的方式来进行类方法(AOP)拦截

1. CGLib作用:

CGLIB(Code Generation Library)是一个开源项目,是一个强大的,高性能,高质量的Code生成类库,它可以在运行期扩展Java类与实现Java接口,通俗说cglib可以在运行时动态生成字节码

2. CGLib的maven地址:


            cglib
            cglib
            3.2.5
            compile

3. 声明一个演示用的类(不是接口),该类有两个构造函数(无参及带参):

public class CGLibBean {
    private String name;
    private String age;
    
    //1. 无参构造函数
    public CGLibBean(){
        this.name = "jacky";
        this.age = "18";
    }
    
    //2. 带参构造函数
    public CGLibBean(String name,String age) {
        this.name = name;
        this.age  = age;
    }

    //3. get/set函数
    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getAge() {
        return age;
    }

    public void setAge(String age) {
        this.age = age;
    }
}

4. 使用CGLib进行类的成员方法拦截:

  • 要用到的包及类
import java.lang.reflect.Method;
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
  • 声明一个类,实现net.sf.cglib.proxy.MethodInterceptor接口,该接口继承自Callback标记接口
public class CGLibProxy implements MethodInterceptor
//空接口
public interface Callback
{
}
public interface MethodInterceptor extends Callback
{
    /**
     * All generated proxied methods call this method instead of the original method.
     * The original method may either be invoked by normal reflection using the Method object,
     * or by using the MethodProxy (faster).
     * @param obj "this", the enhanced object
     * @param method intercepted Method
     * @param args argument array; primitive types are wrapped
     * @param proxy used to invoke super (non-intercepted method); may be called
     * as many times as needed
     * @throws Throwable any exception may be thrown; if so, super method will not be invoked
     * @return any value compatible with the signature of the proxied method. Method returning void will ignore this value.
     * @see MethodProxy
     */    
    public Object intercept(Object obj, java.lang.reflect.Method method, Object[] args,
                               MethodProxy proxy) throws Throwable;

}
  • 实现两个方法,用于代理生成无参和带参的代理对象
public  Object createProxy(Class target){
        Enhancer enhancer = new Enhancer();
        //设置要代理的目标类对象
        enhancer.setSuperclass(target);
        //设置callback
        enhancer.setCallback(this);
        //如果你希望CGLIB创建一个有参数的实例,你应该使用net.sf.cglib.proxy.Enhancer.create(Class[], Object[])。
        //该方法的第一个参数指明参数类型,第二个参数指明参数值。参数中的原子类型需要使用包装类。
        return enhancer.create();
}
public  Object createProxyWithStringParams(Class target){
        Enhancer enhancer = new Enhancer();
        //设置要代理的目标类对象
        enhancer.setSuperclass(target);
        //设置callback
        enhancer.setCallback(this);
        //如果你希望CGLIB创建一个有参数的实例,你应该使用net.sf.cglib.proxy.Enhancer.create(Class[], Object[])。
        //该方法的第一个参数指明参数类型,第二个参数指明参数值。参数中的原子类型需要使用包装类。
        @SuppressWarnings("rawtypes")
        Class[] types = new Class[2];
        types[0] = String.class;
        types[1] = String.class;
        Object[] objs = new String[2];
        objs[0] = "tom";
        objs[1] = "16";
                //生成一个名为tom,年龄16岁的对象
        return enhancer.create(types,objs);
    }
  • 实现MethodInterceptor 接口中的intercept方法
    public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
        System.out.println("before from "+method.getName());
               //比较重要的一个方法,invokeSuper,实际调用的是CGLibBean中的方法
        Object ret = proxy.invokeSuper(obj, args);
        System.out.println("after from "+method.getName());
        return ret;
    }

5. 测试代码:

  • 使用无参函数
public static void testCGLib(){
        CGLibProxy proxy = new CGLibProxy();
        
        CGLibBean obj = (CGLibBean)proxy.createProxy(CGLibBean.class);
        obj.getName();
        obj.getAge();
    }
  • 使用有参函数
public static void testCGLibWithParams() {

    CGLibProxy proxy = new CGLibProxy();
    CGLibBean obj = (CGLibBean) proxy.createProxyWithStringParams(CGLibBean.class);
    obj.getName();
    obj.getAge();
}
  • main函数:
public static void main(String[] args) {
        
        System.out.println("blf test demo");
        testCGLib();   
        System.out.println("---------------------------");
        testCGLibWithParams();
}
  • 显示效果如下:
blf test demo
before from getName
after from getName
before from getAge
after from getAge
---------------------------
before from getName
after from getName
before from getAge
after from getAge

CGLib的缺陷:

不能对final 类型进行aop拦截

到目前为止,我们已经了解了三种AOP的实现方式,各有优缺点。
之所以讲这么多,是为了更好的理解Spring中的AOP
实际Spring中的AOP就是使用了上面提到的那些技术,进行了强大的二次封装
下一篇我们开始关注Spring中的AOP技术

你可能感兴趣的:(Java及Spring框架中的AOP技术汇总--使用CGLib对接口和类进行动态代理及限制条件)