强悍的Spring之AOP CGLIB实现

2021-05-28 11:14:52 浏览数 (1)


1、什么是CGLIB

CGLIB(Code Generator Library)是一个强大的、高性能的代码生成库。其被广泛应用于AOP框架(Spring、dynaop)中,用以提供方法拦截操作。Hibernate作为一个比较受欢迎的ORM框架,同样使用CGLIB来代理单端(多对一和一对一)关联。

2、为什么使用CGLIB

CGLIB代理主要通过对字节码的操作,为对象引入间接级别,以控制对象的访问。我们知道Java中有一个动态代理也是做这个事情的,那我们为什么不直接使用Java动态代理,而要使用CGLIB呢?答案是CGLIB相比于JDK动态代理更加强大,JDK动态代理虽然简单易用,但是其有一个致命缺陷是,只能对接口进行代理。如果要代理的类为一个普通类、没有接口,那么Java动态代理就没法使用了。

3.CGLIB实现代理的原理

首先创建目标对象:

代码语言:javascript复制
public class UserManagerImpl {
    public void addUser(String userName) {
        System.out.println("UserManagerImpl add user name is:"   userName);
    }

    public void deleteUser(String userName) {
        System.out.println("UserManagerImpl delete user name is:"   userName);
    }
}

针对这个目标类,假如我们要使用动态代理实现AOP,那么我们只能在写一个增强的接口,然后让目标类实现增强接口,然后我们就可以使用动态代理实现目标类的增强,可是假如我们不想让目标类实现其他的接口,那么我们就只能使用CGLIB技术来实现目标类的增强了。

CGLIB实现目标类增强的原理是这样的:CGLIB会动态创建一个目标类的子类,然后返回该子类的对象,也就是增强对象,至于增强的逻辑则是在子类中完成的。我们知道子类要么和父类有一样的功能,要么就比父类功能强大,所以CGLIB是通过创建目标类的子类对象来实现增强的,所以:

目标子类 = 目标类 增强逻辑

4.使用CGLIB实现AOP

引入maven坐标:

代码语言:javascript复制
<dependency>
  <groupId>cglib</groupId>
  <artifactId>cglib</artifactId>
  <version>3.2.0</version>
</dependency>

通过CGLIB代理实现如下:

首先实现一个MethodInterceptor,方法调用会被转发到该类的intercept()方法。

然后在需要使用UserManagerImpl(目标对象)的时候,通过CGLIB动态代理获取代理对象。

我们仍然使用上面的UserManagerImpl作为目标对象,然后我们实现一个MethodInterceptor,在实现MethodInterceptor之前,我们先看一下这个接口是什么:

代码语言:javascript复制
public interface MethodInterceptor extends Callback {
    public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable;
}

这里解释一下各个参数的意思:

  • Object为由CGLib动态生成的代理类实例
  • Method为上文中实体类所调用的被代理的方法引用
  • Object[]为参数值列表
  • MethodProxy为生成的代理类对方法的代理引用。

下面是实现的一个MethodInterceptor,同时写了创建增强对象的逻辑即myCglibCreator()方法,该方法返回增强对象。

代码语言:javascript复制
public class CglibFactory implements MethodInterceptor {

    public <T> T myCglibCreator(Class<T> clazz) {
        Enhancer enhancer = new Enhancer();
        //将目标类设置为父类,cglib动态代理增强的原理就是子类增强父类,cglib不能增强目标类为final的类
        enhancer.setSuperclass(clazz);
        //设置回调接口,这里的MethodInterceptor实现类回调接口,而我们又实现了MethodInterceptor
        //这里的回调接口就是本类对象,调用的方法其实就是intercept()方法
        enhancer.setCallback(this);
        //create()方法用于创建cglib动态代理对象
        return (T) enhancer.create();
    }

    //回调接口的方法
    //回调接口的方法执行的条件是:代理对象执行目标方法时会调用回调接口的方法
    @Override
    public Object intercept(Object obj, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
        System.out.println("before action");
        Object result = methodProxy.invokeSuper(obj, args);
        System.out.println("after action");
        //这里实现将返回值字符串变为大写的逻辑
        return Optional.ofNullable(result).map(r -> String.valueOf(r).toUpperCase()).orElse("");
    }
}

其中,Object result = methodProxy.invokeSuper(o, objects);调用代理类实例上的methodProxy方法的父类方法(即实体类UserServiceImpl中对应的方法),然后返回目标方法的返回值result,然后实现增强的逻辑,将返回的字符串变为大写的。下面是测试代码:

代码语言:javascript复制
@Test
public void test_aop() {
    UserManagerImpl proxy = new CglibFactory().myCglibCreator(UserManagerImpl.class);

    String result = proxy.addUser("steven");

    System.out.println("result is:"   result);
}
输出:
before action
UserManagerImpl add user name is:steven
after action
result is:STEVEN

上述代码中,我们通过CGLIB的Enhancer来指定要代理的目标对象、实际处理代理逻辑的对象,最终通过调用create()方法得到代理对象,对这个对象所有非final方法的调用都会转发给MethodInterceptor.intercept()方法,在intercept()方法里我们可以加入任何逻辑,比如修改方法参数,加入日志功能、安全检查功能等;通过调用MethodProxy.invokeSuper()方法,我们将调用转发给原始对象,具体到本例,就是UserServiceImpl的具体方法。CGLIG中MethodInterceptor的作用跟JDK动态代理代理中的InvocationHandler很类似,都是方法调用的中转站。

5、总结

通过CGLIB很方便的实现动态增强,但是CGLIB也有其缺陷,那就是必须目标类必须是可以继承的,如果目标类不可继承,那么我们就无法使用CGLIB来增强该类,我们可以称JDK动态代理实现的AOP为面向接口的动态增强,将CGLIB实现的AOP称为面向子类的动态增强。

主要区别:

  • Java动态代理只能够对接口进行代理,不能对普通的类进行代理(因为所有生成的代理类的父类为Proxy,Java类继承机制不允许多重继承;CGLIB能够代理普通类;
  • Java动态代理使用Java原生的反射API进行操作,在生成类上比较高效;CGLIB使用ASM框架直接对字节码进行操作,在类的执行过程中比较高效

0 人点赞