zoukankan      html  css  js  c++  java
  • Spring 事务注解@Transactional

    事务管理一般有编程式和声明式两种,编程式是直接在代码中进行编写事物处理过程,而声名式则是通过注解方式或者是在xml文件中进行配置,相对编程式很方便。

    而注解方式通过@Transactional 是常见的。我们可以使用@EnableTransactionManagement 注解来启用事务管理功能,该注解可以加在启动类上或者单独加个配置类来处理。

    1、Transactional 注解的属性

    • name 当在配置文件中有多个 TransactionManager , 可以用该属性指定选择哪个事务管理器。
    • propagation 事务的传播行为,默认值为 REQUIRED。
    • isolation 事务的隔离度,默认值采用 DEFAULT。
    • timeout 事务的超时时间,默认值为-1。如果超过该时间限制但事务还没有完成,则自动回滚事务。
    • read-only 指定事务是否为只读事务,默认值为 false;为了忽略那些不需要事务的方法,比如读取数据,可以设置 read-only 为 true。
    • rollback-for 用于指定能够触发事务回滚的异常类型,如果有多个异常类型需要指定,各类型之间可以通过逗号分隔。
    • no-rollback- for 抛出 no-rollback-for 指定的异常类型,不回滚事务。

      propagation 属性(事务传播性)

    •   REQUIRED 支持当前已经存在的事务,如果还没有事务,就创建一个新事务。
    •   MANDATORY 支持当前已经存在的事务,如果还没有事务,就抛出一个异常。
    •   NESTED 在当前事务中创建一个嵌套事务,如果还没有事务,那么就简单地创建一个新事务。
    •   REQUIRES_NEW 挂起当前事务,创建一个新事务,如果还没有事务,就简单地创建一个新事务。
    •   NEVER 强制要求不在事务中运行,如果当前存在一个事务,则抛出异常。
    •   NOT_SUPPORTED 强制不在事务中运行,如果当前存在一个事务,则挂起该事务。
    •   SUPPORTS 支持当前事务,如果没有事务那么就不在事务中运行。

    2、Transactional应用

    @Transactional 可以加在方法上,表示对当前方法配置事务也可以添加到类级别上。

    也可以添加到类级别上。当把@Transactional 注解放在类级别时,表示所有该类的公共方法都配置相同的事务属性信息。

    当类级别配置了@Transactional,方法级别也配置了@Transactional,应用程序会以方法级别的事务属性信息来管理事务,即方法级别的事务属性信息会覆盖类级别的相关配置信息。

    3、Transactional工作原理

    声明式事务管理包含三个组成部分:

    • 事务的切面

    • 事务管理器

    • EntityManager Proxy本身

    事务的切面

    事务的切面是一个“around(环绕)”切面,在注解的业务方法前后都可以被调用。实现切面的具体类是TransactionInterceptor。事务的切面有两个主要职责:

    • 在’before’时,切面提供一个调用点,来决定被调用业务方法应该在正在进行事务的范围内运行,还是开始一个新的独立事务。

    • 在’after’时,切面需要确定事务被提交,回滚或者继续运行。

    在’before’时,事务切面自身不包含任何决策逻辑,是否开始新事务的决策委派给事务管理器完成。

    事务管理器

    事务管理器需要解决下面两个问题:

    • 新的Entity Manager是否应该被创建?

    • 是否应该开始新的事务?

    这些需要事务切面’before’逻辑被调用时决定。事务管理器的决策基于以下两点:

    • 事务是否正在进行

    • 事务方法的propagation属性(比如REQUIRES_NEW总要开始新事务)

    如果事务管理器确定要创建新事务,那么将:

    • 创建一个新的entity manager

    • entity manager绑定到当前线程

    • 从数据库连接池中获取连接

    • 将连接绑定到当前线程

    使用ThreadLocal变量将entity manager和数据库连接都绑定到当前线程。事务运行时他们存储在线程中,当它们不再被使用时,事务管理器决定是否将他们清除。程序的任何部分如果需要当前的entity manager和数据库连接都可以从线程中获取。

    EntityManager proxy

    当业务方法调用类似entityManager.persist()方法时,这不是由entity manager直接调用的,而是业务方法调用代理,因为事物管理器将entity manage绑定到了线程上,代理从线程获取当前的entity manager。

    4、附注

      4.1 @Transactional 注解应用到 public 方法,才能进行事务管理。因为aop会进行拦截是否是public方法:

    //AbstractFallbackTransactionAttributeSource类
      
        protected TransactionAttribute computeTransactionAttribute(Method method,
            Class<?> targetClass) {
                // Don't allow no-public methods as required.
                if (allowPublicMethodsOnly() && !Modifier.isPublic(method.getModifiers())) {
                return null;
           }
      }

      4.2 propagation 属性

      下面三种 propagation 可以不启动事务。错误的配置这三种 propagation,事务可能不会发生回滚。

    • TransactionDefinition.PROPAGATION_SUPPORTS:如果当前存在事务,则加入该事务;如果当前没有事务,则以非事务的方式继续运行。
    • TransactionDefinition.PROPAGATION_NOT_SUPPORTED:以非事务方式运行,如果当前存在事务,则把当前事务挂起。
    • TransactionDefinition.PROPAGATION_NEVER:以非事务方式运行,如果当前存在事务,则抛出异常。

      4.3 rollbackFor 属性
      默认情况下,如果在事务中抛出了未检查异常(继承自 RuntimeException 的异常)或者 Error,则 Spring 将回滚事务;除此之外,Spring 不会回滚事务。
      可以通过rollbackFor来制定在事物中抛出的其他类型的异常来支持事务回滚,例:

    @Transactional(propagation= Propagation.REQUIRED, rollbackFor= MyException.class)

      若在目标方法中抛出的异常是 rollbackFor 指定的异常的子类,事务同样会回滚。

      4.4 在默认的代理模式下,只有目标方法由外部调用,才能被 Spring 的事务拦截器拦截。在同一个类中的两个方法直接调用,是不会被 Spring 的事务拦截器拦截

      像如下这种在同一个类中的两个方法上加上事务控制,其中method上的事务是不能生效的,一种方法就是把它写到另一个列中,然后再当前类中调用

    @Transactional(propagation = Propagation.REQUIRED)
    @Override
    public void save() {
    
        method();
    
        。。。业务处理
    
        if (true) {
            throw new RuntimeException("save 抛异常了");
        }
    }
    
    @Transactional(propagation = Propagation.REQUIRES_NEW)
    public void method() {
        。。。业务处理
    }

    今日话:

      生活或许就是这样吧,忙的时候你想着闲下来,闲下来太久了又想着忙些什么。

    源码参照:Github

  • 相关阅读:
    python全栈开发,Day43(引子,协程介绍,Greenlet模块,Gevent模块,Gevent之同步与异步)
    python全栈开发,Day42(Thread类的其他方法,同步锁,死锁与递归锁,信号量,事件,条件,定时器,队列,Python标准模块--concurrent.futures)
    python全栈开发,Day41(线程概念,线程的特点,进程和线程的关系,线程和python理论知识,线程的创建)
    处理大并发之五 使用libevent利器bufferevent
    [转]./configure,make,make install的作用
    [转]Centos安装zeromq和jzmq
    [zhuan] linux 下 wxWidgets 安装,编译
    [转]面向过程的分析(POA),和面向对象的分析(OOA)
    【转】libevent和基于libevent的网络编程
    [转] Linux下 config/configure/Configure、make 、make test/make check、sudo make install 的作用
  • 原文地址:https://www.cnblogs.com/kingsonfu/p/10413154.html
Copyright © 2011-2022 走看看