Spring详细解读事务管理
作者:羡羡ˇ
什么是事务
事务就是对数据库若干操作组成的一个单元
我们在开发企业应用时,对于业务人员的一个操作实际是对数据读写的多步操作的结合。由于数据操作在顺序执行的过程中,任何一步操作都有可能发生异常,异常会导致后续操作无法完成,此时由于业务逻辑并未正确的完成,之前成功操作数据的并不可靠,需要在这种情况下进行回退
事务的作用就是为了保证用户的每一个操作都是可靠的,事务中的每一步操作都必须成功执行,只要有发生异常就回退到事务开始未进行操作的状态,这些操作要么都完成,要么都取消,从而保证数据满足一致性的要求
如何理解呢 ?
例如 : A现在要转账给B 那么转账是几个方法呢? 两个 : 方法1: A 减钱 方法2: B加钱
如果A方法成功执行后 , B方法中执行时出现了异常, 就等于A钱扣了却没有给B加钱, 那么这样的行为肯定是不允许的, 所以我们引入事务的概念 , 事务一般也是对于数据库而言的
Spring事务配置
Spring事务管理又分为编程式事务和声明式事务
编 程 式 事 务 在 项 目 中 很 少 使 用 , 这 种 方 式 需 要 注 入 一 个 事 务 管 理 对 象 TransactionTemplate ,然后在我们代码中需要提交事务或回滚事务时自己写代码实现
声明式事务管理建立在 AOP 基础上,本质是对方法前后进行拦截,所以声明式事务是方法级别的。
为什么说是基于AOP呢? 因为在通过xml文件配置中我们是这样来做的
<aop:config> <aop:pointcut expression="execution(* com.ff.spring.service.UserService.*(..))" id="allmethod"/> <aop:advisor advice-ref="txadvice" pointcut-ref="allmethod"/> </aop:config>
基于注解的方式直接帮我们省略了这个过程, 更为方便
这里我们主要介绍声明式事务
首先在db.xml中配置spring事务管理类
<!--配置spring事务管理类--> <bean id="sourceTransactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"> <property name="dataSource" ref="druidDataSource"/> </bean>
这里我们直接介绍基于注解方式的spring事务管理
开启注解扫描
<!-- 开启注解事务管理 --> <tx:annotation-driven transaction-manager="sourcetransactionManager"/>
我们使用@Transactiona 这样一个注解标签来声明事务, 它可以作用于方法,表明这个方法支持事务, 也可以作用于类, 表明这个类中的所有方法支持事务
public class UserDao { @Autowired JdbcTemplate jdbcTemplate; @Transactional(propagation= Propagation.REQUIRED) public void saveUser(){ jdbcTemplate.update("insert into admin(account,pass_word,sex) values (?,?,?)","li","111","男"); int i=108/0; //出现异常 jdbcTemplate.update("insert into admin(account,pass_word,sex) values (?,?,?)","qw","111","男"); } }
在上述代码中, 第一条sql 语句虽然成功执行, 但后面出现了异常, 所以这个方法的事务并没有提交, 是不会向数据库提交数据的
Spring事务传播行为
即然是传播,那么至少有两个东西,才可以发生传播。单体不存在传播这个行为。事务传播行为(propagation behavior)指的就是当一个事务方法被另一个事务方法调用时,这个事务方法应该如何进行。事务传播行为是 Spring 框架独有的事务增强特性,他不属于的事务实际提供方数据库行为.
是不是有点懵, 说啥呢这是, 接着来解释
试想 , 有两个方法 A 和 B , 它们都有事务, 那么我在 A 方法中去调用 B 方法, 那么B方法此时
应该怎样去执行呢? 是将B 方法加入到 A 方法组成一个事务,还是它们都是一个独立的事务呢 ?
Spring定义了 7种 事务传播行为, 我们下面主要介绍其中三种,都会举例说明
传播行为一般是对于B方法(被调用的方法而言的)
1. PROPAGATION_REQUIRED
指定的方法必须在事务内执行,若当前存在事务,加入到当前事务中,若当前没有事务,则创建一个新事务,这种传播行为是最常见的,也是 spring 默认的传播行为
A 方法调用 B方法(PROPAGATION_REQUIRED) , 如果A方法是存在事务的, 那么直接将B方
法加入到 A事务中,组成一个事务. 如果 A 方法是没有事务的, 那么B就是一个单独的事务
这么说可能有点绕 , 我们再次请出李雷, 算了, 这次就放过李雷, 换成张三吧
就是说呢 , 我和李雷现在都要去吃饭, 然后我说呢,李雷咱俩一起吧(A调用B,我叫李雷去吃饭) , 不出意外的话(事务顺利执行提交), 我们两吃饭作为一个整体, 最终都吃了饭(AB都存在事务就作为一个整体事务) , 但是还有一种情况, 我叫李雷去吃饭, 但是我却没有去(A调用B, 但是A没有事务),那么这时候李雷肯定单独去吃饭(B单独开启一个事务)
@Transactional(propagation = Propagation.REQUIRED) public void saveDept(){ //A调用B deptDao.saveDept(); commonService.saveLog(); //B }
@Transactional(propagation = Propagation.REQUIRED) public void saveLog(){ commonDao.saveLog(); int a = 10/0; //异常 }
此时 A 调用 B , 它们都有事务, B中出了异常, 那么此时AB最终都不会去和数据库交互
//@Transactional(propagation = Propagation.REQUIRED) public void saveDept(){ //A调用B deptDao.saveDept(); commonService.saveLog(); //B }
@Transactional(propagation = Propagation.REQUIRED) public void saveLog(){ commonDao.saveLog(); int a = 10/0; //异常 }
第二种方式 , 我们将异常加到A 中, 注释掉注解(取消A的事务), 那么此时B是单独的一个事务, B里面出了异常, 不会去和数据库交互, 则A会去和数据库交互
2. PROPAGATION_SUPPORTS
支持当前事务,如果当前没有事务,就以非事务方式执行
A方法调用B方法(PROPAGATION_SUPPORTS), 如果A是存在事务的,那么直接将B方
法加入到 A事务中,组成一个事务. 如果 A 没有事务, 那么B也没有事务
我叫李雷去吃饭 , 如果我一定要去吃饭(A调用B), 那么最终我吃饭和李雷吃饭就总共作为一个事务,就和上面第一个例子是一样的, 作为整体的一个事务. 但是第二种情况就是, 我虽然叫了李雷去吃饭, 但是我最终没有去(A没有事务) , 这时候李雷说, 那我也不去吃饭了(B也没有事务)
@Transactional(propagation = Propagation.REQUIRED) public void saveDept(){ //A调用B deptDao.saveDept(); commonService.saveLog(); //B }
@Transactional(propagation = Propagation.SUPPORTS) public void saveLog(){ commonDao.saveLog(); int a = 10/0; //异常 }
此时 A调用 B , AB都有事务, 那么B 中出现异常, 最终都不会和数据库交互,就和上述第一种情况
一样
//@Transactional(propagation = Propagation.REQUIRED) public void saveDept(){ //A调用B deptDao.saveDept(); commonService.saveLog(); //B }
取消掉A 的事务 , 那么此时B 会以非事务执行 , 这时候AB都会和数据库去交互,因为非事务
3. PROPAGATION_REQUIRES_NEW
总是新建一个事务,如果当前存在事务,把当前事务挂起,直到新建的事务结束。
A方法调用B方法(PROPAGATION_REQUIRES_NEW), 如果A存在事务, 那么B此时会先把A事务挂起, 然后为自己新建一个事务, 先执行完B事务, 才会去执行 A 事务 . 如果A 没有事务, 那么B自己单独新建一个事务执行
这里就是, 我虽然叫李雷去吃饭(A调用B) , 李雷都会先自己去吃饭, 不等我(B自己新建一个事务, A有事务先将A挂起)
@Transactional(propagation = Propagation.REQUIRED) public void saveDept(){ //A调用B deptDao.saveDept(); commonService.saveLog(); //B }
@Transactional(propagation = Propagation.REQUIRES_NEW) public void saveLog(){ commonDao.saveLog(); int a = 10/0; //异常 }
A 调用 B , A支持事务 , 此时B将A事务挂起, 单独开启一个事务,里面出现异常,此时AB都不会和数据库交互
//@Transactional(propagation = Propagation.REQUIRED) public void saveDept(){ //A调用B deptDao.saveDept(); commonService.saveLog(); //B }
取消A的事务, 但此时B事务是独立的, 出现异常, 所以B不会和数据库交互,但是A和数据库交互
声明式事务失效
事务也是会失效的 , 失效有以下几种情况
1.@Transactional 应用在非 public 修饰的方法上
2.@Transactional 注解属性 propagation 设置错误
3.同一个类中方法调用,导致@Transactional 失效
4.异常被 catch 捕获导致@Transactional 失效
5.数据库引擎不支持事务
非public修饰导致权限错误, 事务失效, propagation设置参数错误导致事务失效
同类下方法调用也会导致事务失效
catch如果捕获了异常, 就相当于程序没有异常,这时事务也会失效
最后就是数据库引擎不支持, mysql中只有InnoDB引擎是支持事务的
结语
关于Spring 事务管理就先说到这 , 后面介绍SpringMVC 与 ssm 框架, 谢谢,爱你们😘学 spring 一定要笑, 笑着学 ,嘿嘿
到此这篇关于Spring详细解读事务管理的文章就介绍到这了,更多相关Spring事务管理内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!