事务的回滚五种方法

 

前段时间对Spring的事务配置做了比较深入的研究,在此之间对Spring的事务配置虽说也配置过,但是一直没有一个清楚的认识。通过这次的学习发觉Spring的事务配置只要把思路理清,还是比较好掌握的。

    总结如下:

    Spring配置文件中关于事务配置总是由三个组成部分,分别是DataSource、TransactionManager和代理机制这三部分,无论哪种配置方式,一般变化的只是代理机制这部分。

    DataSource、TransactionManager这两部分只是会根据数据访问方式有所变化,比如使用Hibernate进行数据访问时,DataSource实际为SessionFactory,TransactionManager的实现为HibernateTransactionManager。

    具体如下图:

根据代理机制的不同,总结了五种Spring事务的配置方式,配置文件如下:

    第一种方式:每个Bean都有一个代理


    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:context="http://www.springframework.org/schema/context"
    xmlns:aop="http://www.springframework.org/schema/aop"
    xsi:schemaLocation="http://www.springframework.org/schema/beans
           http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
           http://www.springframework.org/schema/context
           http://www.springframework.org/schema/context/spring-context-2.5.xsd
           http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd">

                class="org.springframework.orm.hibernate3.LocalSessionFactoryBean"> 
         
       
   
 

     
            class="org.springframework.orm.hibernate3.HibernateTransactionManager">
       
   

   
   
   
       
   

   
            class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean"> 
            
               
         
        
         
         
             
                PROPAGATION_REQUIRED
           
 
       
 
   
 

    第二种方式:所有Bean共享一个代理基类


    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:context="http://www.springframework.org/schema/context"
    xmlns:aop="http://www.springframework.org/schema/aop"
    xsi:schemaLocation="http://www.springframework.org/schema/beans
           http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
           http://www.springframework.org/schema/context
           http://www.springframework.org/schema/context/spring-context-2.5.xsd
           http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd">

                class="org.springframework.orm.hibernate3.LocalSessionFactoryBean"> 
         
       
   
 

     
            class="org.springframework.orm.hibernate3.HibernateTransactionManager">
       
   

   
                class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean" 
            lazy-init="true" abstract="true"> 
         
         
         
         
             
                PROPAGATION_REQUIRED 
           
 
       
 
   
   
  
   
   
       
   

   
     
          
   

 

 

 

第三种方式:使用拦截器


    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:context="http://www.springframework.org/schema/context"
    xmlns:aop="http://www.springframework.org/schema/aop"
    xsi:schemaLocation="http://www.springframework.org/schema/beans
           http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
           http://www.springframework.org/schema/context
           http://www.springframework.org/schema/context/spring-context-2.5.xsd
           http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd">

                class="org.springframework.orm.hibernate3.LocalSessionFactoryBean"> 
         
       
   
 

     
            class="org.springframework.orm.hibernate3.HibernateTransactionManager">
       
   
 
  
            class="org.springframework.transaction.interceptor.TransactionInterceptor"> 
         
         
         
             
                PROPAGATION_REQUIRED 
           
 
       
 
   

     
     
         
             
                *Dao
           
 
       
 
         
             
                transactionInterceptor 
           
 
       
 
   
 
 
   
   
       
   

第四种方式:使用tx标签配置的拦截器


    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:context="http://www.springframework.org/schema/context"
    xmlns:aop="http://www.springframework.org/schema/aop"
    xmlns:tx="http://www.springframework.org/schema/tx"
    xsi:schemaLocation="http://www.springframework.org/schema/beans
           http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
           http://www.springframework.org/schema/context
           http://www.springframework.org/schema/context/spring-context-2.5.xsd
           http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd
           http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd">

   
   

                class="org.springframework.orm.hibernate3.LocalSessionFactoryBean"> 
         
       
   
 

     
            class="org.springframework.orm.hibernate3.HibernateTransactionManager">
       
   


   
       
           
       

   

   
   
                    expression="execution(* com.bluesky.spring.dao.*.*(..))" />
                    pointcut-ref="interceptorPointCuts" />       
   
     

 

第五种方式:全注解


    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:context="http://www.springframework.org/schema/context"
    xmlns:aop="http://www.springframework.org/schema/aop"
    xmlns:tx="http://www.springframework.org/schema/tx"
    xsi:schemaLocation="http://www.springframework.org/schema/beans
           http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
           http://www.springframework.org/schema/context
           http://www.springframework.org/schema/context/spring-context-2.5.xsd
           http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd
           http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd">

   
   

   

                class="org.springframework.orm.hibernate3.LocalSessionFactoryBean"> 
         
       
   
 

     
            class="org.springframework.orm.hibernate3.HibernateTransactionManager">
       
   

   

 

 

package com.bluesky.spring.dao;

import java.util.List;

import org.hibernate.SessionFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.orm.hibernate3.support.HibernateDaoSupport;
import org.springframework.stereotype.Component;

import com.bluesky.spring.domain.User;

@Transactional
@Component("userDao")
public class UserDaoImpl extends HibernateDaoSupport implements UserDao {

    public List listUsers() {
        return this.getSession().createQuery("from User").list();
    }
   
   
}

 

 

 

 

 

注意的几点:

1、@Transactional只能被应用到public方法上,对于其他非public的方法,如果标记了@Transactional也不会

报错,但方法没有事务功能。

2、默认情况下,一个有事务方法,遇到RuntimeException时会回滚,遇到受检查的异常是不会回滚的。要想所

有异常都回滚,要加上@Transactional(rollbackFor={Exception.class, 其他异常})。

 

Transactional的所有可选属性如下:

属性                      类型                      默认值         说明

propagation               Propagation枚举           REQUIRED       事务传播属性(下有说明)

isolation                 isolation枚举             DEFAULT        事务隔离级别(另有说明)

readOnly                  boolean                   false          是否只读

timeout                   int                       -1             超时(秒)

rollbackFor               Class[]                   {}             需要回滚的异常类

rollbackForClassName      String[]                  {}             需要回滚的异常类名

noRollbackFor             Class[]                   {}             不需要回滚的异常类

noRollbackForClassName    String[]                  {}             不需要回滚的异常类名

 

事务的隔离级别  有如下可选:

可以去看spring的源代码:org.springframework.transaction.annotation.Isolation

DEFAULT                   采用数据库默认隔离级别

READ_UNCOMMITTED

READ_COMMITTED

REPEATABLE__READ

SERIALIZABLE

 

事务的传播属性,有如下可选:

可以去看spring的源代码:org.springframework.transaction.annotation.Propagation

REQUIRED:业务方法需要在一个事务中运行,如果方法运行时,已处在一个事务中,那么就加入该事务,否则自

          己创建一个新的事物,这是spring默认的传播行为。

SUPPORTS:如果业务方法在某个事物范围内被调用,则方法成为该事务的一部分,如果业务方法在事务范围外被

          调用,则方法在没有事务的环境下执行。

MANDATORY:只能在一个已存在事务中执行,业务方法不能发起自己的事物,如果业务方法在没有事务的环境下

           调用,就抛异常。

REQUIRES_NEW:业务方法总是会为自己发起一个新的事务,如果方法已运行在一个事务中,则原有方法被挂起,

              新的事务被创建,直到方法结束,新事务才结束,原先的事务才会恢复执行。

NOT_SUPPORTED:声明方法需要事务,如果方法没有关联到一个事务,容器不会为他开启事务,如果方法在一个

               事务中被调用,该事务会被挂起,在方法调用结束后,原先的事务便会恢复执行。

NEVER:声明方法绝对不能在事务范围内执行,如果方法在某个事务范围内执行,容器就抛异常,只有没关联到

       事务,才正常执行。

NESTED:如果一个活动的事务存在,则运行在一个嵌套的事务中,如果没有活动的事务,则按REQUIRED属性执

        行。它使用了一个单独的事务,这个事务拥有多个可以回滚的保证点,内部事务回滚不会对外部事务造

        成影响,它只对DataSource。

 

你可能感兴趣的:(事务的回滚五种方法)