A.createA()是一个事务,createA()方法本身会createAentity(),生成A对象。里面包含调用了 B.createB(), C.createC()方法。
B.createB()也是一个事务,包含调用了 B1.createB1(),C1.createC1()方法。B1.createB1会重新从数据库查找A对象,并据此做一些判断来执行后续逻辑。
所以A的createA()事务方法嵌套调用了 B.createB()事务方法,并且B方法内使用了A创建的对象。
问:A,B如何设置事务的传播属性?
事务的传播属性:
PROPAGATION_REQUIRED: 支持当前事务,没有则新建
PROPAGATION_REQUIRESNEW: 新建事务,如果当前存在事务,把当前事务挂起
PROPAGATION_SUPPORTS:支持当前事务,如果当前没有事务,就以非事务方式执行
PROPAGATION_MANDATORY:支持当前事务,如果当前没有事务,就抛出异常
PROPAGATION_NOT_SUPPORTED:以非事务方式执行,如果当前存在事务,就把当前事务挂起。也就是说业务方法不需要事务
PROPAGATION_NEVER:以非事务方式执行,如果当前存在事务,则抛出异常。也就是说业务方法绝对不能在事务范围内执行
PROPAGATION_NESTED:如果一个活动的事务存在,则运行在一个嵌套的事务中。 如果没有活动事务, 则按REQUIRED属性执行
实际上,使用的是前两种:required, requiresNew. 两者区别在于:
PROPAGATION_REQUIRES_NEW 启动一个新的, 不依赖于环境的 "内部" 事务. 这个事务将被完全 commited 或 rolled back 而不依赖于外部事务, 它拥有自己的隔离范围, 自己的锁, 等等. 当内部事务开始执行时, 外部事务将被挂起, 内务事务结束时, 外部事务将继续执行。他是其一个单独的事务,自我完全隔离。
PROPAGATION_REQUIRED 如果当前存在一个事务,则加入当前事务。如果不存在任何事务,则创建一个新的事务。他可能会两个事务合并成一个,或外部没有事务时自己创建一个事务。一般作为事务默认的传播行为。
所以A.createA() 方法使用 REQUIRES_NEW隔离级别;B.createB()方法使用 REQUIRED 隔离级别。因为B需要读取到A先前创建的 AEntity。如果B使用的是REQUIRES_NEW隔离级别,那么将读取不到A创建但尚未commit的AEntity,导致程序错误。