Spring系列第50篇:spring事务拦截器顺序如何控制?

star2017 1年前 ⋅ 286 阅读

1、前言

咱们知道 Spring 事务是通过aop的方式添加了一个事务拦截器,事务拦截器会拦截目标方法的执行,在方法执行前后添加了事务控制。

那么spring事务拦截器的顺序如何控制呢,若我们自己也添加了一些拦截器,此时事务拦截器和自定义拦截器共存的时候,他们的顺序是怎么执行的?如何手动来控制他们的顺序??

可能有些朋友会问,控制他们的顺序,这个功能有什么用呢?为什么要学这个

学会了这些,你可以实现很多牛逼的功能,比如

1、读写分离

2、通用幂等框架

3、分布式事务框架

对这些有兴趣么?那么咱们继续。

2、事务拦截器顺序设置

@EnableTransactionManagement 注解有个 order属性,默认值是Integer.MAX_VALUE,用来指定事务拦截器的顺序,值越小,拦截器的优先级越高,如:

  1. @EnableTransactionManagement(order = 2)

下面来看案例。

3、案例

我们自定义2个拦截器:一个放在事务拦截器之前执行,一个放在事务拦截器之后执行

拦截器 顺序
TransactionInterceptorBefore 1
@EnableTransactionManagement 事务拦截器 2
TransactionInterceptorAfter 3

3.1、准备sql

  1. DROP DATABASE IF EXISTS javacode2018;
  2. CREATE DATABASE if NOT EXISTS javacode2018;
  3. USE javacode2018;
  4. DROP TABLE IF EXISTS t_user;
  5. CREATE TABLE t_user(
  6. id int PRIMARY KEY AUTO_INCREMENT,
  7. name varchar(256) NOT NULL DEFAULT '' COMMENT '姓名'
  8. );

3.2、Spring配置类MainConfig10

@1:开启了事务管理功能,并且设置了事务拦截器的顺序是2,spring事务拦截器完整类名是

  1. org.springframework.transaction.interceptor.TransactionInterceptor

@2:开启aop功能

  1. package com.javacode2018.tx.demo10;
  2. import org.springframework.context.annotation.Bean;
  3. import org.springframework.context.annotation.ComponentScan;
  4. import org.springframework.context.annotation.Configuration;
  5. import org.springframework.context.annotation.EnableAspectJAutoProxy;
  6. import org.springframework.jdbc.core.JdbcTemplate;
  7. import org.springframework.jdbc.datasource.DataSourceTransactionManager;
  8. import org.springframework.transaction.PlatformTransactionManager;
  9. import org.springframework.transaction.annotation.EnableTransactionManagement;
  10. import javax.sql.DataSource;
  11. @Configuration //说明当前类是一个配置类
  12. @ComponentScan //开启bean自动扫描注册功能
  13. @EnableTransactionManagement(order = 2) //@1:设置事务拦截器的顺序是2
  14. @EnableAspectJAutoProxy // @2:开启@Aspect Aop功能
  15. public class MainConfig10 {
  16. @Bean
  17. public DataSource dataSource() {
  18. org.apache.tomcat.jdbc.pool.DataSource dataSource = new org.apache.tomcat.jdbc.pool.DataSource();
  19. dataSource.setDriverClassName("com.mysql.jdbc.Driver");
  20. dataSource.setUrl("jdbc:mysql://localhost:3306/javacode2018?characterEncoding=UTF-8");
  21. dataSource.setUsername("root");
  22. dataSource.setPassword("root123");
  23. dataSource.setInitialSize(5);
  24. return dataSource;
  25. }
  26. //定义一个jdbcTemplate
  27. @Bean
  28. public JdbcTemplate jdbcTemplate(DataSource dataSource) {
  29. return new JdbcTemplate(dataSource);
  30. }
  31. //定义事务管理器transactionManager
  32. @Bean
  33. public PlatformTransactionManager transactionManager(DataSource dataSource) {
  34. return new DataSourceTransactionManager(dataSource);
  35. }
  36. }

3.3、定义一个有事务的Service类

addUser方法上面添加了@Transactional注解,表示使用spring来管理事务,方法内部向db中插入了一条数据,为了方便分析结果,方法内部输出了2行日志

  1. package com.javacode2018.tx.demo10;
  2. import org.springframework.beans.factory.annotation.Autowired;
  3. import org.springframework.jdbc.core.JdbcTemplate;
  4. import org.springframework.stereotype.Component;
  5. import org.springframework.transaction.annotation.Transactional;
  6. @Component
  7. public class UserService {
  8. @Autowired
  9. private JdbcTemplate jdbcTemplate;
  10. @Transactional
  11. public void addUser() {
  12. System.out.println("--------UserService.addUser start");
  13. this.jdbcTemplate.update("insert into t_user(name) VALUES (?)", "张三");
  14. System.out.println("--------UserService.addUser end");
  15. }
  16. }

3.4、自定义第1个拦截器,放在事务拦截器之前执行

下面通过Aspect的方式定义了一个拦截器,顺序通过@Order(1)设置的是1,那么这个拦截器会在事务拦截器之前执行。

  1. package com.javacode2018.tx.demo10;
  2. import org.aopalliance.intercept.Joinpoint;
  3. import org.aspectj.lang.ProceedingJoinPoint;
  4. import org.aspectj.lang.annotation.Around;
  5. import org.aspectj.lang.annotation.Aspect;
  6. import org.aspectj.lang.annotation.Pointcut;
  7. import org.springframework.core.annotation.Order;
  8. import org.springframework.stereotype.Component;
  9. @Component
  10. @Aspect
  11. @Order(1) //@1
  12. public class TransactionInterceptorBefore {
  13. @Pointcut("execution(* com.javacode2018.tx.demo10.UserService.*(..))")
  14. public void pointcut() {
  15. }
  16. @Around("pointcut()")
  17. public Object tsBefore(ProceedingJoinPoint joinPoint) throws Throwable {
  18. System.out.println("--------before start!!!");
  19. Object result = joinPoint.proceed();
  20. System.out.println("--------before end!!!");
  21. return result;
  22. }
  23. }

3.4、自定义第2个拦截器,放在事务拦截器后面执行

这个拦截器的order是3,会在事务拦截器后面执行。

  1. package com.javacode2018.tx.demo10;
  2. import org.aspectj.lang.ProceedingJoinPoint;
  3. import org.aspectj.lang.annotation.Around;
  4. import org.aspectj.lang.annotation.Aspect;
  5. import org.aspectj.lang.annotation.Pointcut;
  6. import org.springframework.core.annotation.Order;
  7. import org.springframework.stereotype.Component;
  8. @Component
  9. @Aspect
  10. @Order(2)
  11. public class TransactionInterceptorAfter {
  12. @Pointcut("execution(* com.javacode2018.tx.demo10.UserService.*(..))")
  13. public void pointcut() {
  14. }
  15. @Around("pointcut()")
  16. public Object tsAfter(ProceedingJoinPoint joinPoint) throws Throwable {
  17. System.out.println("--------after start!!!");
  18. Object result = joinPoint.proceed();
  19. System.out.println("--------after end!!!");
  20. return result;
  21. }
  22. }

3.5、添加测试类

  1. package com.javacode2018.tx.demo10;
  2. import org.junit.Before;
  3. import org.junit.Test;
  4. import org.springframework.context.annotation.AnnotationConfigApplicationContext;
  5. import org.springframework.jdbc.core.JdbcTemplate;
  6. public class Demo10Test {
  7. private UserService userService;
  8. private JdbcTemplate jdbcTemplate;
  9. @Before
  10. public void before() {
  11. AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(MainConfig10.class);
  12. userService = context.getBean(UserService.class);
  13. this.jdbcTemplate = context.getBean("jdbcTemplate", JdbcTemplate.class);
  14. jdbcTemplate.update("truncate table t_user");
  15. }
  16. @Test
  17. public void test1() {
  18. this.userService.addUser();
  19. }
  20. }

3.6、分析test1方法代码执行顺序

咱们先不执行,下分析一下test1方法执行顺序,test1方法内部会调用userService的addUser方法,这个方法会被3个拦截器拦截。

自定义的2个拦截器和事务拦截器TransactionInterceptor拦截,执行顺序如下:

下面来运行一下,看看结果和我们分析的是否一致。

3.7、运行test1输出

  1. --------before start!!!
  2. --------after start!!!
  3. --------UserService.addUser start
  4. --------UserService.addUser end
  5. --------after end!!!
  6. --------before end!!!

结果和上图中一致,大家可以在3个拦截器中设置一下断点,调试一下可以看到更详细的信息,可加深理解。

4、总结

今天的内容算是比较简单的,重点要掌握如何设置事务拦截器的顺序,@EnableTransactionManagement 有个 order属性,默认值是Integer.MAX_VALUE,用来指定事务拦截器的顺序,值越小,拦截器的优先级越高。

后面我们会通过这个功能实现读写分离,通用幂等性的功能。

5、案例源码

  1. git地址:
  2. https://gitee.com/javacode2018/spring-series
  3. 本文案例对应源码:
  4. spring-series\lesson-002-tx\src\main\java\com\javacode2018\tx\demo10

本博客所有系列案例代码以后都会放到这个上面,大家watch一下,可以持续关注动态。

最新资料

更多内容请访问:IT源点

全部评论: 0

    我有话说: