摘抄:EJB3事务管理分两种:CMT(容器管理事务)和BMT(Bean管理事务),SessionBean默认是CMT管理,当然也可以再代码中通过注解显示声明。
@TransactionManagement{TransactionManagementType.CONTAINER}
1,架构与BUG描述
架构采用EJB+ibatis+jboss
beanA使用UserTransaction开始事务,使用中调用beanB,beanB使用ibatis的SqlMapClient执行select for update。接着调用beanC使用ibatis的SqlMapClient执行update同一张表。结果表被锁无法执行。
2,寻找bug始末
1)怀疑BMT方式事务不能跨bean使用,最后研究发现如果使用SqlMapClient.startTransaction确实不能跨bean使用,但是使用ejb的UserTransaction是可以的。因为UserTransaction是将连接和线程放到了一个map,只要是同一个线程,数据库连接肯定是一个。
2)打印了一遍各bean所在的当前线程,均为一个线程。
3)怀疑UserTransaction和ibatis的jdbc连接不是同一个,均为EJB容器管理,应该可以排除
4)后来将JBOSS的连接数改为1,发现在beanA调完beanB之后调beanC的时候报:Unable to get managed connection for /jdbc/ora。说明beanB和beanC用的不同的连接。
5)再自己看代码发现beanC中少了一行@TransactionManagement{TransactionManagementType.BEAN}
3,绕了好大的弯,不过也学了不少事务的知识,以此为记