Spring事务传播行为
传播行为如下(七种形式)
名称 | 含义 | 简介 |
PROPAGATION_REQUIRED | 表示当前方法必须在一个事务中运行。如果存在事务正在进行中,该方法将会在这个事务中运行,否则就会开始一个新的事务进行这个方法 | 有事务就用这个事务,没有就新建事务 |
PROPAGTION_SUPPORTS | 表示当前方法不需要事务性上下文,但是如果有一个事务正在运行,这个方法会在这个事务中运行,没有就以非事务的方式进行 | 有事务就用这个事务,没有就不用事务 |
PROPAGTION_MANDATORY | 表示当前方法必须运行在一个事务中,如果当前方法没有在事务中,将抛出一个异常 | 方法必须运行在一个存在的事务中,否则抛出异常 |
PROPAGATION_REQUIRES_NEW | 表示当前方法必须在他自己开启的事务里运行。会启动一个新的事务,如果存在一个现有的事务,则将会在这个方法运行期间挂起。 | 开启新的事务运行方法,旧事务存在的话就挂起 |
PROPAGATION_NOT_SUPPORTED | 表示该方法不应该在一个事务中运行,如果存在一个现有事务正在进行中,它将在该方法的运行期间被挂起 | 不使用事务,并挂起已存在的事务 |
PROPAGATION_NEVER | 表示当前的方法不应该在一个事务中运行。如果当前有事务正在进行,则会抛出一个异常 | 不使用事务,存在旧事务则抛出异常 |
PROPAGATION_NESTED | 表示如果当前存在一个在进行中的事务,则当前方法应当运行在一个嵌套的事务中。被嵌套的事务可以独立与封装事务进行提交或回滚,如果封装事务不存在,则按照PROPAGATION_REQUIRES进行 | 以嵌套事务进行,如果外部的回滚,则内部也会回滚。外部的提交,内部的也会提交。但是内部事务不会影响外部事务 |
Spring在TransactionDefinition接口中定义了其中事务传播行为
Spring的注解@Transactional,默认式以PROPAGATION_REQUIRED的形式进行的
即有事务就用事务,没有事务就新建
Spring的隔离级别
名称 | 解释 |
ISOLATION_DEFAULT 默认数据库级别 | 这是PlatfromTransactionManager中默认的隔离级别,使用数据库默认的事务隔离级别。其他四个与JDBC的隔离级别相对应 |
ISOLATION_READ_UNCOMMITTED 读未提交 | 事务最低的隔离级别。允许另外一个事务可以看到这个事务未提交的数据。会产生脏读,不可重复读,幻读 |
ISOLATION_READ_COMMITTED 不可重复读 | 保证一个事务修改的数据提交后才会被另外一个事务读取。另外一个事务不能读取该事务未提交的数据 |
ISOLATION_REPEATABLE_READ 可重复读 | 可重复读,保证多次读取同一个数据时,数据和事务开始时的内容是一致的,禁止读取到别的事务未提交的数据。可以防止脏读,不可重复读(重点在修改),但是会出现幻读(重点在增加与删除)。 虽然这种级别无法解决幻读,但是MySQL用一种Next key-lock的算法来实现可重复读并解决幻读。在进行查询时不仅会将当前的操作记录所著,也会将查询范围所著。也就是说,其他事务如果想要在当前事务查询的范围内进行数据操作,是会被阻塞等待获取锁的。 因此MySQL在可重复读的级别下实现了serializable的隔离级别 MySQL的默认级别,可以在数据库中修改 |
ISOLATION_SERIALIZABLE 串行化 | 串行化,不可重复读,最大的亮点是可靠且代价高 所有事务执行(包括查询操作)都会为所处理的数据加锁,操作同一数据的事务都需要等待 可以防止脏读,不可重复读,幻读
|
Spring中的事务分类
- 声明式:通过XML配置或者注解实现(业务逻辑中只管操作数据库,不需要管事务就能自动使用事务)
- 编程式:通过编程代码在业务逻辑时需要自行实现事务
数据库事务
四大特性 ACID
- 原子性(Atomicity):在一个事务中的操作都是一个逻辑(不可分割)的单元,在执行事务序列时,这些操作要么全部成功,要么全部失败
- 一致性(Consistency):事务执行前和执行后,数据的完整性须处于一致的状态
- 隔离性(Isolation):并发事务执行之间是相互隔离的,互不影响
- 持久性(Durability):一旦事务成功提交,对数据库的改变必须是永久的
纯JDBC使用事务如下步骤
- 获取数据库连接Connection c=DriverManager.getConnection()
- 开启事务c.setAutoCommot(true);//自动提交
- 执行CRUD
- 提交/回滚事务c.commit()/c.rollback()
- 关闭连接c.close()