编程式事务
事务功能的相关操作全部通过自己编写代码来实现。
缺点:
- 所有的细节都需要自己完成,比较繁琐。
- 代码重复性过多,相同的代码需要经常重复性的写。
声明式事务
就是将重复性并且有规律的代码进行封装。只需在配置文件中进行简单的配置即可完成操作。
优点:
- 提高开发效率
- 消除冗余代码
- 能够统一管理,方便性能优化等
事务七种传播行为
事务传播行为是为了解决业务层方法之间互相调用的事务问题。
当事务方法被另一个事务方法调用时,必须指定事务应该如何传播。
发生回滚
TransactionDefinition.PROPAGATION_REQUIRED
默认的事务传播行为。如果当前存在事务,则加入该事务;如果当前没有事务,则创建一个新的事务。
TransactionDefinition.PROPAGATION_REQUIRES_NEW
创建一个新的事务,如果当前存在事务,则把当前事务挂起。
TransactionDefinition.PROPAGATION_NESTED
如果当前存在事务,则创建一个事务作为当前事务的嵌套事务来运行;如果当前没有事务,则创建一个新的事务。
TransactionDefinition.PROPAGATION_MANDATORY
如果当前存在事务,则加入该事务;如果当前没有事务,则抛出异常。
不发生回滚
TransactionDefinition.PROPAGATION_SUPPORTS
如果当前存在事务,则加入该事务;如果当前没有事务,则以非事务的方式继续运行。
TransactionDefinition.PROPAGATION_NOT_SUPPORTED
以非事务方式运行,如果当前存在事务,则把当前事务挂起。
TransactionDefinition.PROPAGATION_NEVER
以非事务方式运行,如果当前存在事务,则抛出异常
事务的隔离级别
Serializable(可串行化):事务只能一个接一个地执行,不能并发执行。此隔离级别可有效防止脏读、不可重复读、幻读。
Repeatable Read(可重复读):读取数据的事务将会禁止写事务(但允许读事务),写事务则禁止任何其他事务。此隔离级别可有效防止不可重复读和脏读。
Committed Read(已提交读):读取数据的事务允许其他事务继续访问该行数据,但是未提交的写事务将会禁止其他事务访问该行。此隔离级别可有效防止脏读。
Uncommitted Read(未提交读):如果一个事务已经开始写数据,则另外一个事务不允许同时进行写操作,但允许其他事务读此行数据。此隔离级别可防止丢失更新。
事务的隔离级别越高,越能保证数据库的完整性和一致性。
数据库的默认隔离级别设置为已提交读 (Committed Read),它既能防止脏读,又能有较好的并发性能。虽然这种隔离级别会导致不可重复读、幻读和第二类丢失更新这些并发问题,但能通过采用悲观锁或乐观锁加以控制。
关于@Transactional
当 @Transactional 注解作用于类上时,该类的所有 public 方法将都具有该类型的事务属性。
添加注解后,当这个类里面的方法抛出异常,就会回滚,数据库里面的数据也会回滚。
注意: 在 @Transactional 注解中如果不配置rollbackFor属性,那么事务只会在遇到RuntimeException的时候才会回滚,所以当添加rollbackFor=Exception.class,可以让事务在遇到非运行时异常时也回滚。