一、MethodBeforeAdvice的执行
this.advice.before(mi.getMethod(),mi.getArguments(), mi.getThis() );//调用before方法 return mi.proceed();//执行下一个拦截器
二、AfterReturningAdvice的执行
Object retVal = mi.proceed();//先执行下一个拦截器,如果拦截器已完就执行目标方法 this.advice.afterReturning(retVal, mi.getMethod(),mi.getArguments(), mi.getThis());//最后再执行该方法
这里可以看出before和after的配置顺序和执行顺序的关系
配置的MethodBeforeAdvice会按照配置顺序执行而配置的AfterReturningAdvice则按配置的反序执行
三、Around Advice的执行
Around Advice则提供了一个更加灵活的方式给用户,想实现AfterAdvice则先执行proceed再做自己的操作,想实现Before Advice则先做操作再执行proceed,甚至可以不调用并返回任意值。该实现类的执行顺序则是在配置的顺序上再看是先执行proceed还是先执行自己的操作。
四、ThrowsAdvice的执行
try { return mi.proceed();//执行链往后走,在后续过程中抛出异常则捕获并执行相关处理 } catch (Throwable ex) { Method handlerMethod = getExceptionHandler(ex); if (handlerMethod != null) { invokeHandlerMethod(mi, ex, handlerMethod); } throw ex; }
在以下3种情况时,会首先查找当前ThrowsAdvice拥有的处理该异常或者父类异常的方法
1. 配置中ThrowsAdvice之后的Advice
2. 所有afterAdvice中preceed执行后抛出的异常
3. 目标方法执行异常
下面是执行配置的处理函数
if(method.getParameterTypes().length == 1) {//一个参数的方法 handlerArgs = new Object[] { ex }; } else {//四个参数的方法,method,args,目标对象,异常 handlerArgs = new Object[] {mi.getMethod(), mi.getArguments(),mi.getThis(), ex}; } try { method.invoke(this.throwsAdvice, handlerArgs); } catch (InvocationTargetException targetEx) { throw targetEx.getTargetException(); }
下面是获取处理函数的判断AFTER_THROWING= "afterThrowing"
if (method.getName().equals(AFTER_THROWING) && (method.getParameterTypes().length == 1 ||method.getParameterTypes().length == 4) && Throwable.class.isAssignableFrom(method.getParameterTypes()[method.getParameterTypes().length - 1]) )
这个部分代码可以看出ThrowsAdvice中的两种可用的处理方法
public void afterThrowing(Exception e) public void afterThrowing(Method m,Object[]args,Object target,Exception e)