Spring Aop

Advice通知

  • BeforeAdvice:接口MethodBeforeAdvice,方法before()
  • AfterAdvice:接口AfterReturningAdvice,方法afterReturning()
  • ThrowsAdvice:在抛出异常时回调,这个回调是AOP通过反射机制完成的。

Pointcut切点
概念:决定Advice通知应该作用于哪个连接点。返回一个MethodMatcher,由它来判断是否需要对当前方法调用进行增强,或者是否需要对当前调用方法应用配置好的Advice通知。

  • 通过正则表达式进行匹配
  • 通过方法名进行匹配
  • 等等

Spring AOP的设计和实现

JVM动态代理

这里写图片描述

Advisor通知器
概念:将Advicepointcut结合起来,可以定义应该使用哪个通知在哪个关注点使用。

建立AopProxy代理对象

ProxyFactoryBean:封装了主要代理对象的生成过程,生成方式有两种

  • JDK的Proxy
  • CGLIB


    这里写图片描述

JDK生成AopProxy代理对象

public Obejct getObject(ClassLoader classLoader) {
        if (logger.isDebugEnabled()) {
            logger.debug("Creating JDK dynamic proxy: target source is " + this.advised.getTargetSource());
        }
        Class[] proxiedInterfaces AopProxyUtils.completeProxiedInterfaces(this.advised);
        findDefinedEqualsAndHashCodeMethods(proxiedInterfaces);
        //这是调用JDK生成Proxy的地方
        return Proxy.newProxyInstance(classLoader, proxiedInterfaces, this);
    }

拦截器调用的实现

JdkDynamicAopProxyinvoke拦截
当Proxy对象的代理方法被调用的时候,JdkDynamicAopProxyinvoke方法作为Proxy对象的回调函数被触发,从而通过invoke的具体实现,构造ReflectiveMethodInvocation对象来完成对目标对象方法调用的拦截或者说功能增强。

Cglib2AopProxyintercept拦截
JdkDynamicAopProxy的回调实现非常类似,唯一不同的是构造CglibMethodInvocation对象来完成拦截器链的调用。

对目标方法的调用

  • jdk:使用反射机制得到调用方法的反射对象,然后使用invoke启动对方法反射对象的调用。
  • cglib:通过MethodProxy对象来直接invoke完成的

AOP拦截器链的调用
1. 先进行判断,如果已经运行到拦截器链的末尾,直接调用目标对象的实现方法
2. 否则,沿着拦截器链继续进行,得到下一个拦截器,通过这个拦截器进行matches判断,是否适用于横切增强的场合,如果是,从拦截器中得到通知器,并启动通知器的invoke方法进行切面增强
3. 结束之后,迭代调用proceed方法,直到拦截器链中的拦截器都完成以上的拦截过程为止

配置通知器
MethodBeforeAdviceAdapter的实现

 class MethodBeforeAdviceAdapter implements AdvisorAdapter, Serializable {

    @Override
    public boolean supportsAdvice(Advice advice) {
        return (advice instanceof MethodBeforeAdvice);
    }

    @Override
    public MethodInterceptor getInterceptor(Advisor advisor) {
        MethodBeforeAdvice advice = (MethodBeforeAdvice) advisor.getAdvice();
        return new MethodBeforeAdviceInterceptor(advice);
    }
}

MethodBeforeAdviceInterceptor的实现

public class MethodBeforeAdviceInterceptor implements MethodInterceptor, Serializable {

    private MethodBeforeAdvice advice;

    //为指定的Advice创建对应的MethodBeforeAdviceInterceptor对象
    public MethodBeforeAdviceInterceptor(MethodBeforeAdvice advice) {
        Assert.notNull(advice, "Advice must not be null");
        this.advice = advice;
    }

//这个invoke方法是拦截器的回调方法,会在代理对象的方法被调用时触发回调
    @Override
    public Object invoke(MethodInvocation mi) throws Throwable {
        this.advice.before(mi.getMethod(), mi.getArguments(), mi.getThis() );
        return mi.proceed();
    }
}

AfterReturningAdviceInterceptor的实现

public class AfterReturningAdviceInterceptor implements MethodInterceptor, AfterAdvice, Serializable {

    private final AfterReturningAdvice advice;

    /**
     * Create a new AfterReturningAdviceInterceptor for the given advice.
     * @param advice the AfterReturningAdvice to wrap
     */
    public AfterReturningAdviceInterceptor(AfterReturningAdvice advice) {
        Assert.notNull(advice, "Advice must not be null");
        this.advice = advice;
    }

    @Override
    public Object invoke(MethodInvocation mi) throws Throwable {
        Object retVal = mi.proceed();
        this.advice.afterReturning(retVal, mi.getMethod(), mi.getArguments(), mi.getThis());
        return retVal;
    }

 }

Spring AOP的高级特性

HotSwappableTaregetSource:使用户可以以线程安全的方式切换目标对象,提供所谓的热交换功能。

    public synchronized Object swap(Object newTarget) throws IllegalArgumentException {
        Assert.notNull(newTarget, "Target object must not be null");
        Object old = this.target;
        this.target = newTarget;
        return old;
    }

    public synchronized Object getTarget() {
        return this.target;
    }
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。