Spring事务管理、事务的传播机制
参考:https://juejin.im/post/5b010f27518825426539ba38
Spring事务管理接口:
- PlatformTransactionManager: (平台)事务管理器
- TransactionDefinition: 事务定义信息(事务隔离级别、传播行为、超时、只读、回滚规则)
- TransactionStatus: 事务运行状态
Spring并不直接管理事务,而是提供了多种事务管理器 ,他们将事务管理的职责委托给Hibernate或者JTA等持久化机制所提供的相关平台框架的事务来实现。
Spring事务管理器的接口是: org.springframework.transaction.PlatformTransactionManager ,通过这个接口,Spring为各个平台如JDBC、Hibernate等都提供了对应的事务管理器,但是具体的实现就是各个平台自己的事情了。
比如我们在使用JDBC或者iBatis(就是Mybatis)进行数据持久化操作时,我们的xml配置通常如下:
1 2 3 4 5 6 | <!-- 事务管理器 --> <bean id= "transactionManager" class = "org.springframework.jdbc.datasource.DataSourceTransactionManager" > <!-- 数据源 --> <property name= "dataSource" ref= "dataSource" /> </bean> |
Spring支持两种方式的事务管理:
- 编程式事务管理: 通过Transaction Template手动管理事务,实际应用中很少使用,
- 使用XML配置声明式事务: 推荐使用(代码侵入性最小),实际是通过AOP实现
实现声明式事务的四种方式:
-
基于 TransactionInterceptor 的声明式事务: Spring 声明式事务的基础,通常也不建议使用这种方式,但是与前面一样,了解这种方式对理解 Spring 声明式事务有很大作用。
-
基于 TransactionProxyFactoryBean 的声明式事务: 第一种方式的改进版本,简化的配置文件的书写,这是 Spring 早期推荐的声明式事务管理方式,但是在 Spring 2.0 中已经不推荐了。
-
基于< tx> 和< aop>命名空间的声明式事务管理: 目前推荐的方式,其最大特点是与 Spring AOP 结合紧密,可以充分利用切点表达式的强大支持,使得管理事务更加灵活。
-
基于 @Transactional 的全注解方式: 将声明式事务管理简化到了极致。开发人员只需在配置文件中加上一行启用相关后处理 Bean 的配置,然后在需要实施事务管理的方法或者类上使用 @Transactional 指定事务规则即可实现事务管理,而且功能也不必其他方式逊色。
例子:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 | create table `account` ( `username` varchar ( 99 ), `salary` int ( 11 ) ); insert into `account` (`username`, `salary`) values( '小王' , '3000' ); insert into `account` (`username`, `salary`) values( '小马' , '3000' ); public class OrdersDao { // 注入jdbcTemplate模板对象 private JdbcTemplate jdbcTemplate; public void setJdbcTemplate(JdbcTemplate jdbcTemplate) { this .jdbcTemplate = jdbcTemplate; } // 对数据操作的方法不包含业务操作 /** * 小王少钱的方法 */ public void reduceMoney() { String sql = "update account set salary=salary-? where username=?" ; jdbcTemplate.update(sql, 1000 , "小王" ); } /** * 小马多钱的方法 */ public void addMoney() { String sql = "update account set salary=salary+? where username=?" ; jdbcTemplate.update(sql, 1000 , "小马" ); } } |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 | public class OrdersService { // 注入Dao层对象 private OrdersDao ordersDao; public void setOrdersDao(OrdersDao ordersDao) { this .ordersDao = ordersDao; } // 注入TransactionTemplate对象 private TransactionTemplate transactionTemplate; public void setTransactionTemplate(TransactionTemplate transactionTemplate) { this .transactionTemplate = transactionTemplate; } // 调用dao的方法 // 业务逻辑,写转账业务 public void accountMoney() { transactionTemplate.execute( new TransactionCallback<Object>() { @Override public Object doInTransaction(TransactionStatus status) { Object result = null ; try { // 小马多1000 ordersDao.addMoney(); // 加入出现异常如下面int // i=10/0(银行中可能为突然停电等。。。);结果:小马账户多了1000而小王账户没有少钱 // 解决办法是出现异常后进行事务回滚 int i = 10 / 0 ; // 事务管理配置后异常已经解决 // 小王 少1000 ordersDao.reduceMoney(); } catch (Exception e) { status.setRollbackOnly(); result = false ; System.out.println( "Transfer Error!" ); } return result; } }); } } |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 | <?xml version= "1.0" encoding= "UTF-8" ?> <beans xmlns= "http://www.springframework.org/schema/beans" xmlns:xsi= "http://www.w3.org/2001/XMLSchema-instance" xmlns:context= "http://www.springframework.org/schema/context" xmlns:aop= "http://www.springframework.org/schema/aop" xmlns:tx= "http://www.springframework.org/schema/tx" xsi:schemaLocation="http: //www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd http: //www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd http: //www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd http: //www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd"> <!-- 配置c3po连接池 --> <bean id= "dataSource" class = "com.mchange.v2.c3p0.ComboPooledDataSource" > <!-- 注入属性值 --> <property name= "driverClass" value= "com.mysql.jdbc.Driver" ></property> <property name= "jdbcUrl" value= "jdbc:mysql://localhost:3306/wangyiyun" ></property> <property name= "user" value= "root" ></property> <property name= "password" value= "153963" ></property> </bean> <!-- 编程式事务管理 --> <!-- 配置事务管理器 --> <bean id= "dataSourceTransactionManager" class = "org.springframework.jdbc.datasource.DataSourceTransactionManager" > <!-- 注入dataSource --> <property name= "dataSource" ref= "dataSource" ></property> </bean> <!-- 配置事务管理器模板 --> <bean id= "transactionTemplate" class = "org.springframework.transaction.support.TransactionTemplate" > <!-- 注入真正进行事务管理的事务管理器,name必须为 transactionManager否则无法注入 --> <property name= "transactionManager" ref= "dataSourceTransactionManager" ></property> </bean> <!-- 对象生成及属性注入 --> <bean id= "ordersService" class = "cn.itcast.service.OrdersService" > <property name= "ordersDao" ref= "ordersDao" ></property> <!-- 注入事务管理的模板 --> <property name= "transactionTemplate" ref= "transactionTemplate" ></property> </bean> <bean id= "ordersDao" class = "cn.itcast.dao.OrdersDao" > <property name= "jdbcTemplate" ref= "jdbcTemplate" ></property> </bean> <!-- JDBC模板对象 --> <bean id= "jdbcTemplate" class = "org.springframework.jdbc.core.JdbcTemplate" > <property name= "dataSource" ref= "dataSource" ></property> </bean> </beans> |
(2)基于AspectJ的声明式事务管理
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 | <!-- 配置c3po连接池 --> <bean id= "dataSource" class = "com.mchange.v2.c3p0.ComboPooledDataSource" > <!-- 注入属性值 --> <property name= "driverClass" value= "com.mysql.jdbc.Driver" ></property> <property name= "jdbcUrl" value= "jdbc:mysql://localhost:3306/wangyiyun" ></property> <property name= "user" value= "root" ></property> <property name= "password" value= "153963" ></property> </bean> <!-- 第一步:配置事务管理器 --> <bean id= "dataSourceTransactionManager" class = "org.springframework.jdbc.datasource.DataSourceTransactionManager" > <!-- 注入dataSource --> <property name= "dataSource" ref= "dataSource" ></property> </bean> <!-- 第二步:配置事务增强 --> <tx:advice id= "txadvice" transaction-manager= "dataSourceTransactionManager" > <!-- 做事务操作 --> <tx:attributes> <!-- 设置进行事务操作的方法匹配规则 --> <!-- account开头的所有方法 --> <!-- propagation:事务传播行为; isolation:事务隔离级别; read-only:是否只读; rollback- for :发生那些异常时回滚 timeout:事务过期时间 --> <tx:method name= "account*" propagation= "REQUIRED" isolation= "DEFAULT" read-only= "false" rollback- for = "" timeout= "-1" /> </tx:attributes> </tx:advice> <!-- 第三步:配置切面 切面即把增强用在方法的过程 --> <aop:config> <!-- 切入点 --> <aop:pointcut expression= "execution(* cn.itcast.service.OrdersService.*(..))" id= "pointcut1" /> <!-- 切面 --> <aop:advisor advice-ref= "txadvice" pointcut-ref= "pointcut1" /> </aop:config> |
(3)基于注解的方式
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 | @Transactional (propagation = Propagation.REQUIRED, isolation = Isolation.DEFAULT, readOnly = false , timeout = - 1 ) public class OrdersService { private OrdersDao ordersDao; public void setOrdersDao(OrdersDao ordersDao) { this .ordersDao = ordersDao; } // 调用dao的方法 // 业务逻辑,写转账业务 public void accountMoney() { // 小马多1000 ordersDao.addMoney(); // 加入出现异常如下面int i=10/0(银行中可能为突然停电等。。。);结果:小马账户多了1000而小王账户没有少钱 // 解决办法是出现异常后进行事务回滚 // int i = 10 / 0;// 事务管理配置后异常已经解决 // 小王 少1000 ordersDao.reduceMoney(); } } |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 | <!-- 配置c3po连接池 --> <bean id= "dataSource" class = "com.mchange.v2.c3p0.ComboPooledDataSource" > <!-- 注入属性值 --> <property name= "driverClass" value= "com.mysql.jdbc.Driver" ></property> <property name= "jdbcUrl" value= "jdbc:mysql://localhost:3306/wangyiyun" ></property> <property name= "user" value= "root" ></property> <property name= "password" value= "153963" ></property> </bean> <!-- 第一步:配置事务管理器 (和配置文件方式一样)--> <bean id= "dataSourceTransactionManager" class = "org.springframework.jdbc.datasource.DataSourceTransactionManager" > <!-- 注入dataSource --> <property name= "dataSource" ref= "dataSource" ></property> </bean> <!-- 第二步: 开启事务注解 --> <tx:annotation-driven transaction-manager= "dataSourceTransactionManager" /> <!-- 第三步 在方法所在类上加注解 --> <!-- 对象生成及属性注入 --> <bean id= "ordersService" class = "cn.itcast.service.OrdersService" > <property name= "ordersDao" ref= "ordersDao" ></property> </bean> <bean id= "ordersDao" class = "cn.itcast.dao.OrdersDao" > <property name= "jdbcTemplate" ref= "jdbcTemplate" ></property> </bean> <bean id= "jdbcTemplate" class = "org.springframework.jdbc.core.JdbcTemplate" > <property name= "dataSource" ref= "dataSource" ></property> </bean> |
spring在TransactionDefinition接口中定义了七个事务传播行为:
-
- propagation_requierd:如果当前没有事务,就新建一个事务,如果已存在一个事务中,加入到这个事务中,这是最常见的选择。
- propagation_supports:支持当前事务,如果没有当前事务,就以非事务方法执行。
- propagation_mandatory:使用当前事务,如果没有当前事务,就抛出异常。
- propagation_required_new:新建事务,如果当前存在事务,把当前事务挂起。
- propagation_not_supported:以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。
- propagation_never:以非事务方式执行操作,如果当前事务存在则抛出异常。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· 震惊!C++程序真的从main开始吗?99%的程序员都答错了
· 【硬核科普】Trae如何「偷看」你的代码?零基础破解AI编程运行原理
· 单元测试从入门到精通
· winform 绘制太阳,地球,月球 运作规律
· 上周热点回顾(3.3-3.9)