今天有个朋友问我一个事务不起作用的问题,如果对事务代理不了解的话,这个问题需要引起大家注意。
先看配置文件:
<!-- 配置事务拦截器--> <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回滚的异常范围内。