Spring中的事务管理

矫情吗;* 2022-12-06 12:39 186阅读 0赞

事务的回顾

1、什么是事务?

事务:逻辑上的一组操作,组成这组操作的各个语句,要么全都成功,要么全都失败。


2、事务的特性

原子性:事务不可分割
一致性:事务执行前后数据完整性保持一致
隔离性:一个事务的执行不应该受到其他事务的干扰
持久性:一旦事务结束,数据就持久化到数据库


3、如果不考虑隔离性引发安全性问题

(1)读的问题:
脏读 :一个事务读到另一个事务未提交的数据。
不可重复读 :一个事务读到另一个事务已经提交的update的数据,导致一个事务中多次查询结果不一致。
虚读、幻读 :一个事务读到另一个事务已经提交的insert的数据,导致一个事务中多次查询结果不一致。

(2)写的问题
丢失更新


4、解决读问题

设置事务的隔离级别
Read uncommitted :读未提交,任何读问题解决不了。
Read committed :读已提交,解决脏读,但是不可重复读和虚读有可能发生。
Repeatable read :可重复读,解决脏读和不可重复读,但是虚读有可能发生。
Serializable :串行化执行,解决所有读问题,但同时效率变低了。

Oracle默认使用读已提交,MySQL默认使用可重复读。


Spring中事务管理的API类

1、PlatformTransactionManager:平台事务管理器

平台事务管理器:是一个接口,是Spring用于管理事务的真正的对象。

实现类有:
DataSourceTransactionManager :底层使用JDBC管理事务
HibernateTransactionManager :底层使用Hibernate管理事务


2、TransactionDefinition :事务定义信息

事务定义:用于定义事务的相关的信息,隔离级别、超时信息、传播行为、是否只读。


3、TransactionStatus:事务的状态

事务状态:用于记录在事务管理过程中,事务的状态的对象。


4、事务管理的API之间的关系

Spring进行事务管理的时候,首先平台事务管理器(PlatformTransactionManager)根据事务定义信息(TransactionDefinition)进行事务的管理,在事务管理过程中,产生各种状态,将这些状态的信息记录到事务状态(TransactionStatus)的对象中。


Spring中事务的传播行为

Spring中提供了七种事务的传播行为:分为三类。
只需记住每类的第一个。

1、保证多个操作在同一个事务中

(1)PROPAGATION_REQUIRED :默认值,如果A中有事务,使用A中的事务,如果A没有,创建一个新的事务,将操作AB都包含进来。

(2)PROPAGATION_SUPPORTS :支持事务,如果A中有事务,使用A中的事务。如果A没有事务,不使用事务。

(3)PROPAGATION_MANDATORY :如果A中有事务,使用A中的事务。如果A没有事务,抛出异常。


2、保证多个操作不在同一个事务中

(1)PROPAGATION_REQUIRES_NEW :如果A中有事务,将A的事务挂起(暂停),创建新事务,只包含自身操作。如果A中没有事务,创建一个新事务,只包含自身操作。

(2)PROPAGATION_NOT_SUPPORTED :如果A中有事务,将A的事务挂起。不使用事务管理。

(3)PROPAGATION_NEVER :如果A中有事务,报异常。


3、嵌套式事务

PROPAGATION_NESTED :嵌套事务,如果A中有事务,按照A的事务执行,执行完成后,设置一个保存点,执行B中的操作,如果没有异常,执行通过,如果有异常,可以选择回滚到最初始位置,也可以回滚到保存点。


Spring的事务管理

1、搭建Spring的事务管理的环境

(1)创建Service的接口和实现类

  1. public interface AccountService {
  2. public void transfer(Integer fromId, Integer toId, Double money);
  3. }
  4. public class AccountServiceImpl implements AccountService {
  5. @Resource(name = "accountDao")
  6. private AccountDao accountDao;
  7. @Override
  8. public void transfer(Integer fromId, Integer toId, Double money) {
  9. accountDao.outMoney(fromId, money);
  10. //int i = 1 / 0; // 模拟中间出现异常
  11. accountDao.inMoney(toId, money);
  12. }
  13. }

(2)创建DAO的接口和实现类

  1. public interface AccountDao {
  2. void outMoney(Integer fromId, Double money);
  3. void inMoney(Integer toId, Double money);
  4. }
  5. // 继承JdbcDaoSupport可以获取Jdbc模板,不需要属性注入
  6. public class AccountDaoImpl extends JdbcDaoSupport implements AccountDao {
  7. @Override
  8. public void outMoney(Integer fromId, Double money) {
  9. String sql = "update t_act set balance = balance - ? where id = ?";
  10. this.getJdbcTemplate().update(sql, money, fromId);
  11. }
  12. @Override
  13. public void inMoney(Integer toId, Double money) {
  14. String sql = "update t_act set balance = balance + ? where id = ?";
  15. this.getJdbcTemplate().update(sql, money, toId);
  16. }
  17. }

2、配置Service和DAO:交给Spring管理

  1. <bean id="accountService" class="com.pipi.spring.tx.demo1.AccountServiceImpl"></bean>
  2. <bean id="accountDao" class="com.pipi.spring.tx.demo1.AccountDaoImpl"></bean>

3、配置连接池和JDBC的模板

  1. <!-- 引入jdbc属性配置文件 -->
  2. <context:property-placeholder location="classpath:jdbc.properties" />
  3. <!-- 配置C3P0数据连接池 -->
  4. <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
  5. <property name="driverClass" value="${jdbc.driverClass}" />
  6. <property name="jdbcUrl" value="${jdbc.url}" />
  7. <property name="user" value="${jdbc.username}" />
  8. <property name="password" value="${jdbc.password}" />
  9. </bean>

4、在DAO注入Jdbc的模板:

  1. <bean id="accountDao" class="com.pipi.spring.tx.demo1.AccountDaoImpl">
  2. <!-- 注入dataSource连接池 -->
  3. <property name="dataSource" ref="dataSource" />
  4. </bean>

5、测试

  1. @RunWith(SpringJUnit4ClassRunner.class)
  2. @ContextConfiguration(value = "classpath:com/pipi/spring/tx/demo1/tx1.xml")
  3. public class Spring_tx_Test {
  4. @Resource(name = "accountService")
  5. private AccountService accountService;
  6. @Test
  7. public void test01() {
  8. accountService.transfer(1001, 1002, 500d);
  9. }
  10. }

Spring的事务管理:

第一类:编程式事务(了解,需要手动编写代码)

1、第一步:配置平台事务管理器

  1. <!-- 配置平台事务管理器 -->
  2. <bean id="dataSourceTransactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
  3. <property name="dataSource" ref="dataSource" />
  4. </bean>

2、第二步:Spring提供了事务管理的模板类

  1. <!-- 配置事务的管理的模板 -->
  2. <bean id="transactionTemplate" class="org.springframework.transaction.support.TransactionTemplate">
  3. <property name="transactionManager" ref="dataSourceTransactionManager" />
  4. </bean>

3、第三步:在业务层注入事务管理的模板

  1. <bean id="accountService" class="com.pipi.spring.tx.demo1.AccountServiceImpl">
  2. <property name="accountDao" ref="accountDao" />
  3. <!-- 注入事务管理的模板 -->
  4. <property name="transactionTemplate" ref="transactionTemplate" />
  5. </bean>

4、编写事务管理的代码

  1. @Override
  2. public void transfer(Integer fromId, Integer toId, Double money) {
  3. transactionTemplate.execute(new TransactionCallbackWithoutResult() {
  4. @Override
  5. protected void doInTransactionWithoutResult(TransactionStatus transactionStatus) {
  6. accountDao.outMoney(fromId, money);
  7. int i = 1 / 0; // 模拟中间出现异常
  8. accountDao.inMoney(toId, money);
  9. }
  10. });
  11. }

5、测试:

  1. @Test
  2. public void test01() {
  3. accountService.transfer(1001, 1002, 500d);
  4. }

Spring的事务管理:二类:声明式事务管理(通过AOP配置实现)

1、XML方式的声明式事务管理

(1)第一步:引入aop的开发包
(2)第二步:恢复转账环境,即中间出现异常,钱会转丢。
(3)第三步:配置事务管理器

  1. <!-- 配置事务管理器 -->
  2. <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
  3. <property name="dataSource" ref="dataSource" />
  4. </bean>

(4)第四步:配置增强

  1. <!-- 配置事务的通知增强 -->
  2. <tx:advice id="txAdvice" transaction-manager="transactionManager">
  3. <tx:attributes>
  4. <!-- 事务管理的规则 -->
  5. <!--<tx:method name="save*" propagation="REQUIRED" isolation="DEFAULT" /> <tx:method name="delete*" propagation="REQUIRED" /> <tx:method name="update*" propagation="REQUIRED" /> <tx:method name="find*" read-only="true" />-->
  6. <tx:method name="*" propagation="REQUIRED"/>
  7. </tx:attributes>
  8. </tx:advice>

(5)第五步:AOP的配置

  1. <!-- 配置AOP,切入点与切面,切面用advisor,一个通知与一个切入点的组合 -->
  2. <aop:config>
  3. <aop:pointcut id="pointcut1" expression="execution(* com.pipi.spring.tx.demo2.AccountServiceImpl.*(..))" />
  4. <aop:advisor advice-ref="txAdvice" pointcut-ref="pointcut1" />
  5. </aop:config>

(6)测试程序

  1. @Test
  2. public void test01() {
  3. accountService.transfer(1001, 1002, 500d);
  4. }

2、注解方式的声明式事务管理

(1)第一步:引入aop的开发包
(2)第二步:恢复转账环境
(3)第三步:配置事务管理器

  1. <!-- 配置事务管理器 -->
  2. <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
  3. <property name="dataSource" ref="dataSource" />
  4. </bean>

(4)第四步:开启注解事务

  1. <!-- 开启注解事务 -->
  2. <tx:annotation-driven transaction-manager="transactionManager" />

(5)第五步:在业务层类上添加注解

  1. @Transactional
  2. public class AccountServiceImpl implements AccountService {
  3. // ......
  4. }

(6测试程序的执行:

  1. @Test
  2. public void test01() {
  3. accountService.transfer(1001, 1002, 500d);
  4. }

声明式事务管理的XML方式,与注解方式都有优点:

XML方式的事务管理,能让匹配上的所有类中的方法都使用配置好的事务管理。

而注解方式极其方便,但需要在所有的所需类上添加注解。


发表评论

表情:
评论列表 (有 0 条评论,186人围观)

还没有评论,来说两句吧...

相关阅读

    相关 Spring事务管理

    事务的概念 事务:是数据库操作最基本单位,逻辑上的一组操作,要么都成功,如果有一个失败所有操作都失败。 事务的四大特性: 1. 原子性 2. 一致性 3. 隔离性

    相关 Spring事务管理

    事务的回顾 1、什么是事务? 事务:逻辑上的一组操作,组成这组操作的各个语句,要么全都成功,要么全都失败。 -------------------- 2、事务的特性

    相关 Spring事务管理

    一、什么是事务 事务就是一系列的动作, 它们被当做一个单独的工作单元. 这些动作要么全部完成, 要么全部不起作用。例如网上购物:购物者的账户减少相应的数额和商品库存减少相应

    相关 spring事务管理(三)

    spring中的事务管理(三) 事务的隔离级别         从理论上来说, 事务应该彼此完全隔离, 以避免并发事务所导致的问题. 然而,那样会对性能产生极大的

    相关 Spring事务管理

    事务的概念 事务代表逻辑上的一组操作要么一起成功, 要么一起失败. 可以想一下转账的例子. 事务的特性 ACID 原子性: 事务是不可分割的. 一致性: