前置通知:在切入点方法执行之前执行
后置通知:在切入点方法正常执行之后执行,它和异常通知永远只能执行一个
异常通知:在切入点方法执行产生异常之后执行,它和后置通知永远只能执行一个
最终通知:无论切入点方法是否正常执行,它都会在其后面执行
Logger类:
package net.togogo.utils;
/**
* 记录日志的工具类,它里面提供了公共代码
*/
public class Logger {
/**
* 前置通知
*/
public void beforePrintLog(){
System.out.println("前置通知Logger类中的beforePrintLog方法开始记录日志了。。。");
}
/**
* 后置通知
*/
public void afterReturningPrintLog(){
System.out.println("后置通知Logger类中的afterReturningPrintLog方法开始记录日志了。。。");
}
/**
* 异常通知
*/
public void afterThrowingPrintLog(){
System.out.println("异常通知Logger类中的afterThrowingPrintLog方法开始记录日志了。。。");
}
/**
* 最终通知
*/
public void afterPrintLog(){
System.out.println("最终通知Logger类中的afterPrintLog方法开始记录日志了。。。");
}
}
bean.xml
<!--配置AOP-->
<aop:config>
<!--配置切面-->
<aop:aspect id="logAdvice" ref="logger">
<!--配置前置通知:在切入点方法执行之前执行-->
<aop:before method="beforePrintLog"
pointcut="execution(* net.togogo.service.impl.*.*(..))"></aop:before>
<!--配置后置通知:在切入点方法正常执行之后执行,它和异常通知永远只能执行一个-->
<aop:after-returning method="afterReturningPrintLog"
pointcut="execution(* net.togogo.service.impl.*.*(..))"></aop:after-returning>
<!--配置异常通知:在切入点方法执行产生异常之后执行,它和后置通知永远只能执行一个-->
<aop:after-throwing method="afterThrowingPrintLog"
pointcut="execution(* net.togogo.service.impl.*.*(..))"></aop:after-throwing>
<!--配置最终通知:无论切入点方法是否正常执行,它都会在其后面执行-->
<aop:after method="afterPrintLog"
pointcut="execution(* net.togogo.service.impl.*.*(..))"></aop:after>
</aop:aspect>
</aop:config>
配置切入点表达式<aop:pointcut>
<!--配置AOP-->
<aop:config>
<!--配置切面-->
<aop:aspect id="logAdvice" ref="logger">
<!--配置切入点表达式:id属性用于指定表达式的唯一标识,expression属性用于指定表达式内容
此标签写在aop:aspect标签内部只能当前切面使用
它还可以写在aop:aspect标签外面,此时就变成了所有切面可用
注意:写在aop:aspect标签外面时,要写在最前面!!
-->
<aop:pointcut id="pt" expression="execution(* net.togogo.service.impl.*.*(..))"/>
<!--配置前置通知:在切入点方法执行之前执行-->
<aop:before method="beforePrintLog" pointcut-ref="pt"></aop:before>
<!--配置后置通知:在切点方法正常执行之后执行,它和异常通知永远只能执行一个-->
<aop:after-returning method="afterReturningPrintLog" pointcut-ref="pt"></aop:after-returning>
<!--配置异常通知:在切入点方法执行产生异常之后执行,它和后置通知永远只能执行一个-->
<aop:after-throwing method="afterThrowingPrintLog" pointcut-ref="pt"></aop:after-throwing>
<!--配置最终通知:无论切入点方法是否正常执行,它都会在其后面执行-->
<aop:after method="afterPrintLog" pointcut-ref="pt"></aop:after>
</aop:aspect>
</aop:config>
此标签写在aop:aspect标签内部只能当前切面使用。
它还可以写在aop:aspect标签外面,此时就变成了所有切面可用
注意:写在aop:aspect标签外面时,要写在最前面!!
<!--配置AOP-->
<aop:config>
<!--配置切入点表达式:id属性用于指定表达式的唯一标识,expression属性用于指定表达式内容
此标签写在aop:aspect标签内部只能当前切面使用
它还可以写在aop:aspect标签外面,此时就变成了所有切面可用
注意:写在aop:aspect标签外面时,要写在最前面!!
-->
<aop:pointcut id="pt" expression="execution(* net.togogo.service.impl.*.*(..))"/>
<!--配置切面-->
<aop:aspect id="logAdvice" ref="logger">
<!--配置前置通知:在切入点方法执行之前执行-->
<aop:before method="beforePrintLog" pointcut-ref="pt"></aop:before>
<!--配置后置通知:在切点方法正常执行之后执行,它和异常通知永远只能执行一个-->
<aop:after-returning method="afterReturningPrintLog" pointcut-ref="pt"></aop:after-returning>
<!--配置异常通知:在切入点方法执行产生异常之后执行,它和后置通知永远只能执行一个-->
<aop:after-throwing method="afterThrowingPrintLog" pointcut-ref="pt"></aop:after-throwing>
<!--配置最终通知:无论切入点方法是否正常执行,它都会在其后面执行-->
<aop:after method="afterPrintLog" pointcut-ref="pt"></aop:after>
</aop:aspect>
</aop:config>
环绕通知:
问题:
当我们在baen.xml中配置了环绕通知之后,切入点方法没有执行,而通知方法执行了。
分析:
通过对比动态代理中的环绕通知代码,发现动态代理的环绕通知有明确的切入点方法调用,而我们的代码中没有。
解决:
Spring框架为我们提供了一个接口:ProceedingJoinPoint。该接口有一个方法proceed(),此方法就相当于明确调用切入点方法。该接口可以作为环绕通知的方法参数,在程序执行时,spring框架会为我们提供该接口的实现类供我们使用。
spring中的环绕通知:
他是spring框架为我们提供的一种可以在代码中手动控制增强方法何时执行方式。
/**
* 环绕通知
*/
public Object aroundPrintLog(ProceedingJoinPoint proceedingJoinPoint){
Object rtValue = null;
try {
Object[] args = proceedingJoinPoint.getArgs();//得到方法执行所需参数
System.out.println("aroundPrintLog方法开始记录日志了。。。前置");
rtValue = proceedingJoinPoint.proceed(args);//明确调用业务层方法(切入点方法)
System.out.println("aroundPrintLog方法开始记录日志了。。。后置");
} catch (Throwable throwable) {
System.out.println("aroundPrintLog方法开始记录日志了。。。异常");
throwable.printStackTrace();
}finally {
System.out.println("aroundPrintLog方法开始记录日志了。。。最终");
return rtValue;
}
}
<!--配置AOP-->
<aop:config>
<!--配置切面-->
<aop:aspect id="logAdvice" ref="logger">
<!--配置切入点表达式:id属性用于指定表达式的唯一标识,expression属性用于指定表达式内容
此标签写在aop:aspect标签内部只能当前切面使用
它还可以写在aop:aspect标签外面,此时就变成了所有切面可用
注意:写在aop:aspect标签外面时,要写在最前面!!
-->
<aop:pointcut id="pt" expression="execution(* net.togogo.service.impl.*.*(..))"/>
<aop:around method="aroundPrintLog" pointcut-ref="pt"></aop:around>
</aop:aspect>
</aop:config>