什么导致spring事务失效

今天有个朋友问我一个事务不起作用的问题,如果对事务代理不了解的话,这个问题需要引起大家注意。

先看配置文件:

 <!-- 配置事务拦截器-->
    <bean id="transactionInterceptor"
          class="org.springframework.transaction.interceptor.TransactionInterceptor"> <!-- 事务拦截器bean需要依赖注入一个事务管理器 -->
        <property name="transactionManager" ref="myTransactionManager"/>
        <property name="transactionAttributes"> <!-- 下面定义事务传播属性-->
            <props>
                <!--del开头的方法都被事物管理-->
                <prop key="update*">PROPAGATION_REQUIRED</prop>
                <prop key="add*">PROPAGATION_REQUIRED</prop>
                <prop key="set*">PROPAGATION_REQUIRED</prop>
                <prop key="del*">PROPAGATION_REQUIRED</prop>
                <prop key="find*">PROPAGATION_REQUIRED,readOnly</prop>
            </props>
        </property>
    </bean>
    <!-- 定义BeanNameAutoProxyCreator,该bean无需被引用,因此没有id属性,这个bean根据事务拦截器为目标bean自动创建事务代理-->
    <bean class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator"><!-- 指定对满足哪些bean name的bean自动生成业务代理 -->
        <property name="beanNames"> <!-- 下面是所有需要自动创建事务代理的bean-->
            <list>
                <value>*Service</value>
                <!-- buy开头的bean都会被自动代理-->
            </list>
            <!-- 此处可增加其他需要自动创建事务代理的bean-->
        </property>
        <!-- 下面定义BeanNameAutoProxyCreator所需的事务拦截器-->
        <property name="interceptorNames">
            <list>
                <value>transactionInterceptor</value>
                  <!-- 此处可增加其他新的Interceptor -->
            </list>
        </property>
    </bean>

下面是具体代码:

  @Override
    public boolean addActionSet(ActionEntity actionEntity, ActioncolumnEntity ae, String type) {
        try {
            ae.setMenuflag(type);
            ae.setViewmode("1");
            ae.setIsexpand("1");
            int dbid = this.actionSetDao.add(ae);//这里是第一个增加方法
            actionEntity.setActioncolumnid(dbid);
            actionEntity.setPid(ae.getFatherid());
            this.actionSetDao.add(actionEntity);//这里是第二个增加方法
            loginService.loadAllUserCache(userCache.getUserEntity());
            return true;
        } catch (Exception e) {
            logger_run.error("增加权限异常{}", e);
        }
        return false;
    }

当第一个增加成功时,因为数据库字段长度过小导致第一个增加方法没有成功执行,后台抛出以下异常:

org.springframework.dao.DataIntegrityViolationException: Could not execute JDBC batch update; 
  SQL [insert into frame.action (actioncolumnid, pid, datation, action) values (?, ?, ?, ?)];
  nested exception is org.hibernate.exception.DataException: Could not execute JDBC batch update

 

Caused by: java.sql.BatchUpdateException: Data truncation: Data too long for column 'action' at row 1

预想是数据回滚,两个数据库表都没有执行成功,但是结果是第一个方法成功执行,数据已被插入。

分析原因:

  用BeanNameAutoProxyCreator配置事务代理,完全可以避免增量式配置,所有的事务代理由系统自动创建。容器中的目标bean自动消失,避免需要使用嵌套bean来保证目标bean不可被访问。

  TranscationInterceptor是一个事务拦截器bean,需要传入一个TransactionManager的引用。配置中使用 Spring依赖注入该属性,事务拦截器的事务属性通过transactionAttributes来指定,该属性有props子元素,配置文件中定义了三个事务传播规则:
  所有以insert开始的方法,采用PROPAGATION_REQUIRED的事务传播规则。程序抛出 MyException异常及其子异常时,自动回滚事务。所有以find开头的方法,采用PROPAGATION_REQUIRED事务传播规则,并且只读。其他方法,则采用PROPAGATION_REQUIRED的事务传播规则。
BeanNameAutoProxyCreator是个根据 bean名生成自动代理的代理创建器,该bean通常需要接受两个参数。第一个是beanNames属性,该属性用来设置哪些bean需要自动生成代理。另一个属性是interceptorNames,该属性则指定事务拦截器,自动创建事务代理时,系统会根据这些事务拦截器的属性来生成对应的事务代理。

解决办法:

    去掉addActionSet方法中的try .. catch。

 注:Spring中对异常的回滚,默认是在抛出运行时异常(RuntimeException)时才回滚,对非运行时异常不回滚。如果使用 -Exception,意思是对所有的异常异常都回滚。Exception前面加上 "-" 时,表示发生指定异常时撤消操作(rollback),如果前面加上 "+",表示发生异常时立即提交(commit)。

其他遇到事物失效的情况也有可能是抛出的异常不在spring回滚的异常范围内。 

posted @ 2012-07-17 13:59  王 庆  阅读(2715)  评论(0编辑  收藏  举报