zoukankan      html  css  js  c++  java
  • 什么导致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回滚的异常范围内。 

  • 相关阅读:
    POJ-2955 Brackets(括号匹配问题)
    NYOJ
    石子合并问题
    hdu 4915 括号匹配+巧模拟
    hdu 4920
    hdu 4911 求逆序对数+树状数组
    hdu 4923 单调栈
    hdu 4930 斗地主恶心模拟
    hdu 4927 组合+公式
    hdu 4925 黑白格
  • 原文地址:https://www.cnblogs.com/zhishan/p/2595078.html
Copyright © 2011-2022 走看看