写点什么

MyBatis 事务管理

作者:Java高工P7
  • 2021 年 11 月 11 日
  • 本文字数:2282 字

    阅读完需:约 7 分钟

为不同的事务 API 提供一致的编程模型,比如 JTA(Java Transaction API), JDBC, Hibernate, JPA(Java Persistence API 和 JDO(Java Data Objects)支持声明式事管理,特别是基于注解的声明式事务管理,简单易用提供比其他事务 API 如 JTA 更单的编程式事务管理 API 与 spring 数据访问抽象的完美集成。


二、事务管理方式




spring 支持编程式事务管理和声明式事务管理两种方式。


编程式事务管理使用TransactionTemplate或者直接使用底层的PlatformTransactionManager。对于编程式事务管理,spring 推荐使用TransactionTemplate


声明式事务管理建立在 AOP 之上的。其本质是对方法前后进行拦截,然后在目标方法开始之前创建或者加入一个事务,在执行完目标方法之后根据执行情况提交或者回滚事务。声明式事务最大的优点就是不需要通过编程的方式管理事务,这样就不需要在业务逻辑代码中掺杂事务管理的代码,只需在配置文件中做相关的事务规则声明(或通过基于@Transactional注解的方式),便可以将事务规则应用到业务逻辑中。


显然声明式事务管理要优于编程式事务管理,这正是 spring 倡导的非侵入式的开发方式。声明式事务管理使业务代码不受污染,一个普通的 POJO 对象,只要加上注解就可以获得完全的事务支持。和编程式事务相比,声明式事务唯一不足地方是,后者的最细粒度只能作用到方法级别,无法做到像编程式事务那样可以作用到代码块级别。但是即便有这样的需求,也存在很多变通的方法,比如,可以将需要进行事务管理的代码块独立为方法等等。


声明式事务管理也有两种常用的方式,一种是基于 tx 和 aop 名字空间的 xml 配置文件,另一种就是基于@Transactional注解。显然基于注解的方式更简单易用,更清爽。


三、spring 事务特性




spring 所有的事务管理策略类都继承自org.springframework.transaction.PlatformTransactionManager接口


public interface PlatformTransactionManager {


TransactionStatus getTransaction(TransactionDefinition definition)


throws TransactionException;


void commit(TransactionStatus status) throws TransactionException;


void rollback(TransactionStatus status) throws TransactionException;


}


除了指定不同的事务管理器之后,还能对事务进行隔离级别和传播行为的控制,下面分别详细解释:


四、隔离级别




隔离级别是指若干个并发的事务之间的隔离程度,与我们开发时候主要相关的场景包括:脏读取、重复读、幻读。


对于脏读、不可重复读和幻读详见:何为脏读、不可重复读、幻读


我们可以看org.springframework.transaction.annotation.Isolation枚举类中定义了五个表示隔离级别的值:


public enum Isolation {


DEFAULT(-1),


READ_UNCOMMITTED(1),


READ_COMMITTED(2),


REPEATABLE_READ(4),


SERIALIZABLE(8);


}


  • DEFAULT:这是默认值,表示使用底层数据库的默认隔离级别。对大部分数据库而言,通常这值就是:READ_COMMITTED。

  • READ_UNCOMMITTED:该隔离级别表示一个事务可以读取另一个事务修改但还没有提交的数据。该级别不能防止脏读和不可重复读,因此很少使用该隔离级别。

  • READ_COMMITTED:该隔离级别表示一个事务只能读取另一个事务已经提交的数据。该级别可以防止脏读,这也是大多数情况下的推荐值。

  • REPEATABLE_READ:该隔离级别表示一个事务在整个过程中可以多次重复执行某个查询,并且每次返回的记录都相同。即使在多次查询之间有新增的数据满足该查询,这些新增的记录也会被忽略。该级别可以防止脏读和不可重复读。

  • SERIALIZABLE:所有的事务依次逐个执行,这样事务之间就完全不可能产生干扰,也就是说,该级别可以防止脏读、不可重复读以及幻读。但是这将严重影响程序的性能。通常情况下也不会用到该级别。


指定方法:通过使用 isolation 属性设置,例如:


@Transactional(isolation = Isolation.DEFAULT)


五、传播行为




所谓事务的传播行为是指,如果在开始当前事务之前,一个事务上下文已经存在,此时有若干选项可以指定一个事务性方法的执行行为。


我们可以看org.springframework.transaction.annotation.Propagation枚举类中定义了 6 个表示传播行为的枚举值:


public enum Propagation {


REQUIRED(0),


SUPPORTS(1),


MANDATORY(2),


REQUIRES_NEW(3),


NOT_SUPPORTED(4),


NEVER(5),


NESTED(6);


}


  • REQUIRED:如果当前存在事务,则加入该事务;如果当前没有事务,则创建一个新的事务。

  • SUPPORTS:如果当前存在事务,则加入该事务;如果当前没有事务,则以非事务的方式继续运行。

  • MANDATORY:如果当前存在事务,则加入该事务;如果当前没有事务,则抛出异常。

  • REQUIRES_NEW:创建一个新的事务,如果当前存在事务,则把当前事务挂起。

  • NOT_SUPPORTED:以非事务方式运行,如果当前存在事务,则把当前事务挂起。

  • NEVER:以非事务方式运行,如果当前存在事务,则抛出异常。

  • NESTED:如果当前存在事务,则创建一个事务作为当前事务的嵌套事务来运行;如果当前没有事务,则该取值等价于 REQUIRED。


【一线大厂Java面试题解析+后端开发学习笔记+最新架构讲解视频+实战项目源码讲义】
浏览器打开:qq.cn.hn/FTf 免费领取
复制代码


指定方法:通过使用 propagation 属性设置,例如:


@Transactional(propagation = Propagation.REQUIRED)


六、事务超时




所谓事务超时,就是指一个事务所允许执行的最长时间,如果超过该时间限制但事务还没有完成,则自动回滚事务。在 TransactionDefinition 中以 int 的值来表示超时时间,其单位是秒。


默认设置为底层事务系统的超时值,如果底层数据库事务系统没有设置超时值,那么就是 none,没有超时限制。


七、事务只读属性




只读事务用于客户代码只读但不修改数据的情形,只读事务用于特定情景下的优化,比如使用 Hibernate 的时候。


默认为读写事务。

用户头像

Java高工P7

关注

还未添加个人签名 2021.11.08 加入

还未添加个人简介

评论

发布
暂无评论
MyBatis事务管理