zoukankan      html  css  js  c++  java
  • Spring事务基本配置及传播属性简介

    从Spring 事务配置说起:

      先看看Spring 事务的基础配置

    <aop:aspectj-autoproxy proxy-target-class="true"/>
      <bean id="transactionManager"
        class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <property name="dataSource" ref="dataSource"/>
      </bean>
      <tx:annotation-driven transaction-manager="transactionManager"/>
      <!-- 配置事务传播特性-->
      <tx:advice id="transactionAdvice" transaction-manager="transactionManager">
        <tx:attributes>
          <tx:method name="add*" propagation="REQUIRED"
          rollback-for="Exception,RuntimeException,SQLException"/>
          <tx:method name="remove*" propagation="REQUIRED"
            rollback-for="Exception,RuntimeException,SQLException"/>
          <tx:method name="modify*" propagation="REQUIRED"rollback-for="Exception,RuntimeException,SQLException"/>
          <tx:method name="login" propagation="NOT_SUPPORTED"/>
          <tx:method name="query*" read-only="true"/>
        </tx:attributes>
      </tx:advice>
      <aop:config>
        <aop:pointcut expression="execution(public * com.gupaoedu.vip..*.service..*Service.*(..))"
        id="transactionPointcut"/>
      <aop:advisor pointcut-ref="transactionPointcut" advice-ref="transactionAdvice"/>
    </aop:config>

       Spring 事务管理基于AOP 来实现,主要是统一封装非功能性需求。

    数据库事务原理详解:

    1、事务基本概念

      事务(Transaction)是访问并可能更新数据库中各种数据项的一个程序执行单元(unit)。特点:事务是恢复和并发控制的基本单位。事务应该具有4 个属性:原子性、一致性、隔离性、持久性。这四个属性通常称为ACID 特性。

    • 原子性(Automicity)。一个事务是一个不可分割的工作单位,事务中包括的诸操作要么都做,要么都不做。
    • 一致性(Consistency)。事务必须是使数据库从一个一致性状态变到另一个一致性状态。一致性与原子性是密切相关的。
    • 隔离性(Isolation)。一个事务的执行不能被其他事务干扰。即一个事务内部的操作及使用的数据对并发的其他事务是隔离的,并发执行的各个事务之间不能互相干扰。
    • 持久性(Durability)。持久性也称永久性(Permanence),指一个事务一旦提交,它对数据库中数据的改变就应该是永久性的。接下来的其他操作或故障不应该对其有任何影响。

    2、事务的基本原理

      Spring 事务的本质其实就是数据库对事务的支持,没有数据库的事务支持,Spring 是无法提供事务功能的。对于纯JDBC 操作数据库,想要用到事务,可以按照以下步骤进行:

    1. 获取连接Connection con = DriverManager.getConnection()
    2. 开启事务con.setAutoCommit(true/false);
    3. 执行CRUD
    4. 提交事务/回滚事务con.commit() / con.rollback();
    5. 关闭连接conn.close();

      使用Spring 的事务管理功能后,我们可以不再写步骤2 和4 的代码,而是由Spirng 自动完成。 那么Spring 是如何在我们书写的CRUD 之前和之后开启事务和关闭事务的呢?解决这个问题,也就可以从整体上理解Spring 的事务管理实现原理了。下面简单地介绍下,注解方式为例子配置文件开启注解驱动,在相关的类和方法上通过注解@Transactional 标识。Spring 在启动的时候会去解析生成相关的bean,这时候会查看拥有相关注解的类和方法,并且为这些类和方法生成代理,并根据@Transaction 的相关参数进行相关配置注入,这样就在代理中为我们把相关的事务处理掉了(开启正常提交事务,异常回滚事务)。真正的数据库层的事务提交和回滚是通过binlog 或者redo log 实现的。

    3、Spring 事务的传播属性

      所谓spring 事务的传播属性,就是定义在存在多个事务同时存在的时候,spring 应该如何处理这些事务的行为。这些属性在TransactionDefinition 中定义,具体常量的解释见下表:

    常量名称 常量解释
    PROPAGATION_REQUIRED

    支持当前事务,如果当前没有事务,就新建一个事务。

    这是最常见的选择,也是Spring默认的事务的传播

    PROPAGATION_REQUIRES_NEW

    新建事务,如果当前存在事务,把当前事务挂起。

    新建的事务将和被挂起的事务没有任何关系,是两个独立的事务,

    外层事务失败回滚之后,不能回滚内层事务执行的结果,

    内层事务失败抛出异常,外层事务捕获,也可以不处理回滚操作

    PROPAGATION_SUPPORTS

    支持当前事务,如果当前没有事务,就以非事务方式执行。

    PROPAGATION_MANDATORY

    支持当前事务,如果当前没有事务,就抛出异常。

    PROPAGATION_NOT_SUPPORTED

    以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。

    PROPAGATION_NEVER

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

    PROPAGATION_NESTED

    如果一个活动的事务存在,则运行在一个嵌套的事务中。

    如果没有活动事务,则按REQUIRED 属性执行。

    它使用了一个单独的事务,这个事务拥有多个可以回滚的保存点。

    内部事务的回滚不会对外部事务造成影响。

    它只对DataSourceTransactionManager 事务管理器起效。

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

    4、数据库隔离级别

    隔离级别 隔离级别的值 导致的问题
    Read-Uncommitted 0 导致脏读
    Read-Committed 1 避免脏读,允许不可重复读和幻读
    Repeatable-Read 2 避免脏读,不可重复读,允许幻读
    Serializable 3 串行化读,事务只能一个一个执行,避免了
    脏读、不可重复读、幻读。执行效率慢,使
    用时慎重

     

     

     

     

     

     

     

     

     

    脏读:一事务对数据进行了增删改,但未提交,另一事务可以读取到未提交的数据。如果第一个事务这时候回滚了,那么第二个事务就读到了脏数据。

    不可重复读:一个事务中发生了两次读操作,第一次读操作和第二次操作之间,另外一个事务对数据进行了修改,这时候两次读取的数据是不一致的。

    幻读:第一个事务对一定范围的数据进行批量修改,第二个事务在这个范围增加一条数据,这时候第一个事务就会丢失对新增数据的修改。

      总结:隔离级别越高,越能保证数据的完整性和一致性,但是对并发性能的影响也越大。大多数的数据库默认隔离级别为Read Commited,比如SqlServer、Oracle少数数据库默认隔离级别为:Repeatable Read 比如: MySQL InnoDB

    5、Spring 中的隔离级别

    常量 解释
    ISOLATION_DEFAULT

    这是个PlatfromTransactionManager 默认的隔离级别,

    使用数据库默认的事务隔离级别。另外四个与JDBC 的隔离级别相对应。

    ISOLATION_READ_UNCOMMITTED

    这是事务最低的隔离级别,它允许另外一个事务可以看到这个事务未提交的数据。

    这种隔离级别会产生脏读,不可重复读和幻像读。

    ISOLATION_READ_COMMITTED

    保证一个事务修改的数据提交后才能被另外一个事务读取。

    另外一个事务不能读取该事务未提交的数据。

    ISOLATION_REPEATABLE_READ

    这种事务隔离级别可以防止脏读,不可重复读。但是可能出现幻像读.

    ISOLATION_SERIALIZABLE

    这是花费最高代价但是最可靠的事务隔离级别。事务被处理为顺序执行。

     

     

     

     

     

     

     

     

     

     

     

     

     

     

    6、Spring 事务API 架构图:

     

      关键类

    public interface PlatformTransactionManager {
        TransactionStatus getTransaction(
                TransactionDefinition definition) throws TransactionException;
        void commit(TransactionStatus status) throws TransactionException;
        void rollback(TransactionStatus status) throws TransactionException;
    }

      事务真正的开始、提交、回滚都是通过PlatformTransactionManager这个接口来实现的,例如,我们常用的org.springframework.jdbc.datasource.DataSourceTransactionManager。TransactionDefinition用于获取事务的一些属性,Isolation, Propagation,Timeout,Read-only,还定义了事务隔离级别,传播属性等常量。TransactionStatus用于设置和查询事务的状态,如是否是新事务,是否有保存点,设置和查询RollbackOnly等。

       了解更多的事务源码实现原理跳转:https://www.cnblogs.com/wuzhenzhao/p/12869784.html

  • 相关阅读:
    字典与集合
    gitee
    在使用pycharm时同时缩进、左移、多行注释
    代码1(while循环和IF条件语句,字符格式化,break,continue)
    python基础-工具
    11 Serializer组件
    10 响应模块
    09 异常模块
    08 解析模块
    07 渲染模块
  • 原文地址:https://www.cnblogs.com/wuzhenzhao/p/10966549.html
Copyright © 2011-2022 走看看