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