Spring事务

1、事务不生效场景

1、spring团队建议在具体的方法或者类上面使用事务,而不要在类要实现的接口上面。接口上使用 @Transactional 注解,只能当你设置了基于接口的代理时它才生效。因为注解是 不能继承 的

2、数据库引擎设置不对造成的。比如我们最常用的mysql,引擎MyISAM,是不支持事务操作的。需要改成InnoDB才能支持

3、事务的入口方法必须是public,否则不会生效(主要是spring的aop特性决定),另外private、final、static方法使用事务也不会生效

4、spring的事务默认是在运行时自动处理,如果手动catch掉是不会回滚的

5、业务和事务操作必须在一个线程内,否则是不会生效的,如:

    @Transactional
    public void save(String id) {
        new Thread(() -> save(id)).start();
    }

6、检查类是否被代理了(因为spring的事务实现原理为AOP,只有通过代理对象调用方法才能被拦截,事务才能生效)

7、无事务方法调用有事务方法也不会生效

2、声明式事务@Transactional

属性 类型 描述
value String 可选的限定描述符,指定使用的事务管理器
propagation enum: Propagation 可选的事务传播行为设置
isolation enum: Isolation 可选的事务隔离级别设置
readOnly boolean 读写或只读事务,默认读写
timeout int (in seconds granularity) 事务超时时间设置
rollbackFor Class对象数组,必须继承自Throwable 导致事务回滚的异常类数组
rollbackForClassName 类名数组,必须继承自Throwable 导致事务回滚的异常类名字数组
noRollbackFor Class对象数组,必须继承自Throwable 不会导致事务回滚的异常类数组
noRollbackForClassName 类名数组,必须继承自Throwable 不会导致事务回滚的异常类名字数组

 @Transactional 可以作用于接口、接口方法、类以及类方法上。当作用于类上时,该类的所有 public 方法将都具有该类型的事务属性。

事务超时 timeout

为了保证应用程序更好的执行,它的事务不能运行太多时间,因事务会对后端数据库造成锁定,长时间的运行事务将会不必要的占用数据库资源,可以给事务设置一个超时时间,在特定时间后将会自动回滚,不需要等到事务结束

 

3、 事务隔离级别

在一个典型的应用程序中,多个事务同时运行,经常会为了完成他们的工作而操作同一个数据。并发虽然是必需的,但是会导致一下问题:

  • 脏读(Dirty read)-- 脏读发生在一个事务读取了被另一个事务改写但尚未提交的数据时。如果这些改变在稍后被回滚了,那么第一个事务读取的数据就会是无效的。
  • 不可重复读(Nonrepeatable read)-- 不可重复读发生在一个事务执行相同的查询两次或两次以上,但每次查询结果都不相同时。这通常是由于另一个并发事务在两次查询之间更新了数据。
  • 幻读(Phantom reads)-- 幻影读和不可重复读相似。当一个事务(T1)读取几行记录后,另一个并发事务(T2)插入了一些记录时,幻影读就发生了。在后来的查询中,第一个事务(T1)就会发现一些原来没有的额外记录。

 

隔离级别 含义
ISOLATION_DEFAULT 使用后端数据库默认的隔离级别。
ISOLATION_READ_UNCOMMITTED 允许读取尚未提交的更改。可能导致脏读、幻影读或不可重复读。
ISOLATION_READ_COMMITTED 允许从已经提交的并发事务读取。可防止脏读,但幻影读和不可重复读仍可能会发生。
ISOLATION_REPEATABLE_READ 对相同字段的多次读取的结果是一致的,除非数据被当前事务本身改变。可防止脏读和不可重复读,但幻影读仍可能发生。
ISOLATION_SERIALIZABLE 完全服从ACID的隔离级别,确保不发生脏读、不可重复读和幻影读。这在所有隔离级别中也是最慢的,因为它通常是通过完全锁定当前事务所涉及的数据表来完成的。

在@Transactional中事务的五个隔离级别被归类到枚举类Isolation.Isolation的属性值:

  • default:使用基础数据存储的默认隔离级别
  • read_uncommitted:读未提交,允许一个事务可以操作另一个无提交的事务,允许脏读、不可重复读、幻读,隔离级别最低,并发性能最高 
  • read_committed:读未提交,一个事务不可以操作另外一个未提交的事务,能防止脏读,允许不可重复读和幻读
  • repeatable_read:可重复读,表示脏读和不可重复的读取被阻止;可以发生幻读
  • serializable:序列化,表示脏读、不可重复读、幻读被阻止

 

4、事务传播级别

  • propagation_required:如果一个事务存在,则支持当前事务,如果不存在,则创建新的事务
  • propagation_supports:如果一个事务存在,则支持当前事务,如果不存在,则非事务的方法运行
  • propagation_mendatory:如果一个事务存在,则支持当前事务,如果存在,则抛出异常
  • propagation_requires_new:总是要开启一个新的事务,如果事务存在,将该事务挂起
  • propagation_not_supported:总是非事务方法运行,并挂起所有的事务
  • propagation_never:总是非事务方法运行,如果事务存在则抛出异常
  • propagation_nested:某一个事务存在,则运行在一个嵌套的事务中

你可能感兴趣的:(JAVA,spring)