haidaoxianzi 2020-01-12
目录
上一节学习了spring aop的基本概念和如何基于xml配置来实现aop功能。这一节来学习下如何用注解实现aop
上节中虽然也使用的是注解+xml,但只是把bean的配置用注解来实现,aop相关的切面,切入点等配置都是在spring的xml配置文件中实现的。
<!--通知类使用注解配置--> <!--使用aop:config 来声明aop的配置--> <aop:config> <aop:aspect id="transactionAspect" ref="transactionManager"> <!--这个切入点表达式表示com.lyy.service及其子包中的任意类的任意方法--> <aop:pointcut id="servicePointcut" expression="execution(* com.lyy.service..*.*(..))"/> <!--在切面的内部配置通知的类型--> <!--配置前置通知--> <aop:before method="startTransaction" pointcut-ref="servicePointcut"/> <!--后置通知--> <aop:after-returning method="commit" pointcut-ref="servicePointcut"/> <!--配置异常通知--> <aop:after-throwing method="rollBack" pointcut-ref="servicePointcut"/> </aop:aspect> </aop:config>
这一节我们首先来实现把aop的配置也用注解来实现。
在配置文件中开启spring对注解aop的支持,
<!--开启spring对注解aop的支持--> <aop:aspectj-autoproxy/>
在通知类上写上@Aspect
注解,把这个类声明为切面类
在通知类中定义一个方法,打上@Pointcut
注解,说明这个方法用来定义切入点表达式,方法名代表表达式的名称
//用来定义切入点表达式的方法,方法名就是切入点表达式的名称 @Pointcut("execution(* com.lyy.service..*.*(..))") public void pt1(){ }
在对应的方法名上打上通知类型对应的注解,定义通知
/** * 通知类 */ @Component @Aspect//把当前类声明为切面类 public class TransactionManager { @Autowired private ConnectionUtils connectionUtils; @Before("pt1()") public void startTransaction(){ Connection connection = connectionUtils.getConnection(); try { System.out.println("开启事务"); connection.setAutoCommit(false); } catch (SQLException e) { e.printStackTrace(); throw new RuntimeException("开启事务失败,"+e.toString()); } } /** * 提交事务 */ @AfterReturning("pt1()")//后置通知 public void commit(){ Connection connection = connectionUtils.getConnection(); try { System.out.println("提交事务"); connection.commit(); } catch (SQLException e) { e.printStackTrace(); throw new RuntimeException("提交事务失败,"+e.toString()); } } /** * 回滚事务 */ @AfterThrowing("pt1()")//异常通知 public void rollBack(){ Connection connection = connectionUtils.getConnection(); try { System.out.println("业务异常,事务回滚"); connection.rollback(); } catch (SQLException e) { e.printStackTrace(); throw new RuntimeException("回滚事务失败,"+e.toString()); } } /** * spring aop的环绕通知,手动控制增强代码的执行时机 * @param proceedingJoinPoint * @return */ public Object around(ProceedingJoinPoint proceedingJoinPoint){ Object returnValue=null; try { //开启事务 startTransaction(); Object[] args = proceedingJoinPoint.getArgs(); returnValue = proceedingJoinPoint.proceed(args); //提交事务 commit(); } catch (Throwable throwable) { throwable.printStackTrace(); //回滚事务 rollBack(); } finally { //最终通知 } return returnValue; } //用来定义切入点表达式的方法,方法名就是切入点表达式的名称 @Pointcut("execution(* com.lyy.service..*.*(..))") public void pt1(){ }
首先需要在配置类上用一个注解@EnableAspectJAutoProxy
开启对注解aop的支持,然后其余的步骤和上边的2、3、4相同。
多个aop切入同一个方法时,默认的执行顺序是随机的,可以通过配置order来设置执行顺序
在配置切面时加上order属性
<aop:aspect ref="aopBean" order="0">
在切面类上加@Order
注解
注意order越小,顺序越早
先执行的aop方法一定后结束
可以理解为spring aop就是一个同心圆,要执行的方法为圆心,最外层的order最小。从最外层按照AOP1、AOP2的顺序依次执行doAround方法,doBefore方法。然后执行method方法,最后按照AOP2、AOP1的顺序依次执行doAfter、doAfterReturn方法。也就是说对多个AOP来说,先before的,一定后after
示例工程