Spring的奥秘AOP(三)

简介: 前文讲到增强某个类的功能可以提取公共接口,让代理类实现接口达到增强的目的。想想是不是还有其他方式呢? 对了面向对象三大特性之一-----继承,通过继承来增强父类的功能,这也就是CGLIB实现AOP的原理,Spring AOP也采用了这种方案。

1、什么是CGLIB


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


2、为什么使用CGLIB


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


3.CGLIB实现代理的原理


首先创建目标对象:

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坐标:

<dependency>
  <groupId>cglib</groupId>
  <artifactId>cglib</artifactId>
  <version>3.2.0</version>
</dependency>
复制代码

通过CGLIB代理实现如下:


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

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


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


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()方法,该方法返回增强对象。

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方法的父类方法(即实体类SomeService中对应的方法),然后返回目标方法的返回值result,然后实现增强的逻辑,将返回的字符串变为大写的。下面是测试代码:

@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()方法,我们将调用转发给原始对象,具体到本例,就是SomeService的具体方法。CGLIG中MethodInterceptor的作用跟JDK动态代理代理中的InvocationHandler很类似,都是方法调用的中转站。


5、总结


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


主要区别:


  • Java动态代理只能够对接口进行代理,不能对普通的类进行代理(因为所有生成的代理类的父类为Proxy,Java类继承机制不允许多重继承;CGLIB能够代理普通类;


  • Java动态代理使用Java原生的反射API进行操作,在生成类上比较高效;CGLIB使用ASM框架直接对字节码进行操作,在类的执行过程中比较高效


相关文章
|
1月前
|
监控 Java 开发者
Spring AOP动态代理
Spring AOP动态代理
46 1
|
1月前
|
Java Spring 容器
Spring的AOP失效场景详解
Spring的AOP失效场景详解
119 0
|
2月前
|
XML Java 编译器
Spring AOP初步理解及使用
Spring AOP初步理解及使用
49 0
|
2月前
|
Java Spring
[Spring]aop的配置与使用
[Spring]aop的配置与使用
40 0
[Spring]aop的配置与使用
|
1月前
|
设计模式 Java Maven
Spring Aop 底层责任链思路实现-springaopdi-ceng-ze-ren-lian-si-lu-shi-xian
Spring Aop 底层责任链思路实现-springaopdi-ceng-ze-ren-lian-si-lu-shi-xian
36 1
|
2月前
|
XML Java 数据格式
5个点轻松搞定Spring AOP底层实现原理
AOP 也是 Spring 中一个较为重要的内容,相对于传统的 OOP 模式,AOP 有很多让人难以理解的地方,本篇文章将向大家介绍 AOP 的实现方法及其底层实现,内容包括:
47 1
|
18天前
|
XML 监控 Java
Spring AOP:解锁切面编程的威力与实践
Spring AOP:解锁切面编程的威力与实践
21 0
Spring AOP:解锁切面编程的威力与实践
|
27天前
|
XML Java Maven
Spring之Aop的注解使用
Spring之Aop的注解使用
|
1月前
|
Java Spring
Spring 如何实现 AOP
Spring 如何实现 AOP
17 0
|
1月前
|
Java 编译器 程序员
Spring AOP 和 AspectJ 的比较
Spring AOP 和 AspectJ 的比较
39 0