Spring学习总结(五)——事务管理

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/qq_40348465/article/details/85018908

1.事务的基本概念

     事务是一组操作的执行单元,对于数据库操作来说,事务管理的是一组SQL指令(比如增加,修改,删除等)。事务的要求一致性,就是说事务内的操作要么全部执行成功,要么出现了差错,哪怕只有一条SQL语句没有执行成功,那么这一组操作都将全部回滚。    

 2.事务的四大特性(ACID)

 (1)原子性(Atomicity):事务是一个原子操作,由一系列动作组成。事务的原子性确保动作要么全部完成,要么完全不起作用。

 (2)一致性(Consistency):一旦事务完成(不管成功还是失败),系统必须确保它所建模的业务处于一致的状态,而不会是部分完成部分失败。在现实中的数据不应该被破坏。

   (3)隔离性(Isolation):可能有许多事务会同时处理相同的数据,因此每个事务都应该与其他事务隔离开来,防止数据损坏。

 (4)持久性(Durability):一旦事务完成,无论发生什么系统错误,它的结果都不应该受到影响,这样就能从任何系统崩溃中恢复过来。通常情况下,事务的结果被写到持久化存储器中。

3.事务的传播行为

    事务传播行为就是多个事务方法调用时,如何定义方法间事务的传播,当事务方法被另一个事务方法调用时,必须指定事务应该如何传播。Spring中定义了七种传播行为:

传播行为 含义
PROPAGATION_REQUIRED 表示当前方法必须运行在事务中。如果当前事务存在,方法将会在该事务中运行。否则,会启动一个新的事务
PROPAGATION_SUPPORTS 表示当前方法不需要事务上下文,但是如果存在当前事务的话,那么该方法会在这个事务中运行
PROPAGATION_MANDATORY 表示该方法必须在事务中运行,如果当前事务不存在,则会抛出一个异常
PROPAGATION_REQUIRED_NEW 表示当前方法必须运行在它自己的事务中。一个新的事务将被启动。如果存在当前事务,在该方法执行期间,当前事务会被挂起。如果使用JTATransactionManager的话,则需要访问TransactionManager
PROPAGATION_NOT_SUPPORTED 表示该方法不应该运行在事务中。如果存在当前事务,在该方法运行期间,当前事务将被挂起。如果使用JTATransactionManager的话,则需要访问TransactionManager
PROPAGATION_NEVER 表示当前方法不应该运行在事务上下文中。如果当前正有一个事务在运行,则会抛出异常
PROPAGATION_NESTED 表示如果当前已经存在一个事务,那么该方法将会在嵌套事务中运行。嵌套的事务可以独立于当前事务进行单独地提交或回滚。如果当前事务不存在,那么其行为与PROPAGATION_REQUIRED一样。注意各厂商对这种传播行为的支持是有所差异的。可以参考资源管理器的文档来确认它们是否支持嵌套事务

4.事务的隔离级别 

  DEFAULT:使用后端数据库默认的隔离级别(spring中的的选择项)

  READ_UNCOMMITED:允许你读取还未提交的改变了的数据。可能导致脏、幻、不可重复读

  READ_COMMITTED:允许在并发事务已经提交后读取。可防止脏读,但幻读和 不可重复读仍可发生

  REPEATABLE_READ:对相同字段的多次读取是一致的,除非数据被事务本身改变。可防止脏、不可重复读,但幻读仍可能发生。

  SERIALIZABLE:完全服从ACID的隔离级别,确保不发生脏、幻、不可重复读。这在所有的隔离级别中是最慢的,它是典型的通过完全锁定在事务中涉及的数据表来完成的。

补充:

   (1)脏读:指当一个事务正字访问数据,并且对数据进行了修改,而这种数据还没有提交到数据库中,这时,另外一个事务也访问这个数据,然后使用了这个数据。因为这个数据还没有提交那么另外一个事务读取到的这个数据我们称之为脏数据。依据脏数据所做的操作肯能是不正确的。
   (2)不可重复读:指在一个事务内,多次读同一数据。在这个事务还没有执行结束,另外一个事务也访问该同一数据,那么在第一个事务中的两次读取数据之间,由于第二个事务的修改第一个事务两次读到的数据可能是不一样的,这样就发生了在一个事物内两次连续读到的数据是不一样的,这种情况被称为是不可重复读。
    (3)幻象读:一个事务先后读取一个范围的记录,但两次读取的纪录数不同,我们称之为幻象读(两次执行同一条 select 语句会出现不同的结果,第二次读会增加一数据行,并没有说这两次执行是在同一个事务中)。幻读是指当事务不是独立执行时发生的一种现象,例如第一个事务对一个表中的数据进行了修改,比如这种修改涉及到表中的"全部数据行"。同时,第二个事务也修改这个表中的数据,这种修改是向表中插入"一行新数据"。那么,以后就会发生操作第一个事务的用户发现表中还存在没有修改的数据行,就好象发生了幻觉一样.一般解决幻读的方法是增加范围锁RangeS,锁定检索范围为只读,这样就避免了幻读。

      大多数据库默认的隔离级别为Read Commited,如(SqlServer,Oracle)少部分数据库默认的隔离级别Repeatable_Read,如(Mysql)。

5.Spring提供的两种事务的管理方式

    编程式事务管理声明式事务管理 。

    二者的区别:程式事务允许用户在代码中精确定义事务的边界,而声明式事务(基于AOP)有助于用户将操作与事务规则进行解耦。 简单地说,编程式事务侵入到了业务代码里面,但是提供了更加详细的事务管理;而声明式事务由于基于AOP,所以既能起到事务管理的作用,又可以不影响业务代码的具体实现。

 6.Spring所支持的几种事务管理器

blob.png 在单一的JDBC Datasource中的管理事务
org.springframework.orm.hibernate4.HibernateTransactionManager 当持久化机制是hibernate时,用它来管理事务
org.springframework.jdo.JdoTransactionManager 当持久化机制是Jdo时,用它来管理事务
org.springframework.transaction.jta.JtaTransactionManager blob.png
org.springframework.orm.ojb.PersistenceBrokerTransactionManager 当apache的ojb用作持久化机制时,用它来管理事务

 7.基于XML的声明式事务的配置示例

  (1)配置事务管理器bean:使用DataSourceTransactionManager

    

<bean id="txManage" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
    <property name="dataSource" ref="dataSource"></property>
</bean>

   (2)配置需要事务管理器拦截(通知)的方法:使用(*)通配符。如下面的配置需要拦截的方法的规则是save开头的方法和delete开头的方法。

    

<tx:advice id="txAdvice" transaction-manager="txManager">
  <tx:attributes>
    <tx:method name="save*"/>
    <tx:method name="delete*"/>
  </tx:attributes>
</tx:advice>

   (3)配置AOP拦截业务逻辑层的所有方法,通知txAdvice进行拦截, execution:拦截的路径匹配

  

<aop:config>
    <aop:advisor advice-ref="txAdvice" pointcut="execution(* com.transactionManagerStudy..*Service..*(..))"/>
</aop:config>

小结:上面的声明式事务管理的配置的目的是使用AOP的技术对service中的需要事务控制的方法进行拦截,然后通知事务管理器进行事务处理。

8.基于注解的方式进行事务配置示例

  (1)同XML配置一样,需要配置事务管理器 

    

<bean id="txManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
      <property name="dataSource" ref="dataSource"></property>
</bean>

  (2)取代AOP的是,需要配置事务注解驱动

<tx:annotation-driven transaction-manager="txManager"/>

   

 (3)在使用事务的方法上进行标注

  

@Transactional
  public boolean saveOrderInfo(OrderInfo info) {
}

  

猜你喜欢

转载自blog.csdn.net/qq_40348465/article/details/85018908