Spring Mybaties事务的困惑!求解!!

Spring对service事务的配置

tx:attributes









<!--规定当前方法支持当前事务处理,但如果没有事务在运行就使用非事务方法执行 -->
/tx:attributes
/tx:advice
CustomerService有Add()方法,
public Customer add(Customer t) {
customerDao.save(t);
if(customerQuery.getCustomerName().equals("AAAA")){
throw new NullPointerException("到底回滚不?");
}
t.setCustomerName("Super Test");
customerDao.update(t);
return t;

}
问题:如果增加一个Customer,customerName属性是"BBBB",那么数据库会增加一条customerName属性是"Super Test"的数据,
如果增加一个Customer,customerName属性是"AAAA",以我的理解既然add的事务级别是propagation="SUPPORTS",那么在抛出异常之前应该已经保存了“AAAA”数据,但实际情况是没有保存数据直接回滚了,
这是为什么呢?


SUPPORTS声明业务方法不需要事务,如果业务方法执行时已经在一个事务中,则事务被挂起,等方法执行完毕后,事务恢复进行;所以无论你的方法会不会抛出异常,都不会有事务的操作,也就是你对数据库的更新操作都不会生效,因为没有事务,自然不会有事务的提交了,至于回滚就更谈不上了。

customerName等于AAAA的时候,你If语句块里面都直接抛出异常了,
下面的代码就不走了,update方法都没走,你哪来的事务操作?哪来的回滚?

你的事务管理器应该是配到service这一层的吧,这要save update这两个方法用的是同一个事务了啊,那肯定要么同事成功,要么同时失败。

[size=x-small][color=red]我来救你吧[/color][/size]

首先你需要把你定义的事物范围贴出来,这么关键的东西不贴出来,别人也能帮你分析,我真的很惊讶。

假设你的范围是这样定义的。(假设定义到你的CustomerService这个类上面)

也就是你的add方法已经是一个被代理的方法了,SPRING会把事物上下文放到当前线程里面去,然后进入你的add方法,然后里面所有针对数据库的操作底层都是一个connection,所以无论在这个方法上面地方抛出异常都必将会滚

首先你要明白事务REQUIRED是当前有事务就加入,没就新建一个,你这save与update在同一事务里,所以都回滚。
要想达到你想的那就效果(save ROLLBACK, update COMMIT),就新开事务,给update配成REQUIRES_NEW,放在2个service里。