JPA中的事务使用多个线程

时间:2012-10-25 13:53:02

标签: java spring java-ee jpa

我的目标是使用单元测试在我的应用程序中激发一个乐观的锁定异常。我已经理解了如何在理论上做到这一点。但我在实践中的问题是如何在两个线程之间维护一个事务?

所以,这是我到目前为止所做的:

我正在使用JUnit测试:

@RunWith(SpringJUnit4ClassRunner.class)

使用EntityManager org.springframework.orm.jpa.JpaTransactionManager

其中定义了每个方法@Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)

并以entityManager.getTransaction().begin();开始交易,以entityManager.getTransaction().rollback());

结束

这在单线程测试中非常有用,可以保存,更新等。

要创建多个线程,我使用Springs TaskExecutor(类似于这里所描述的内容:Any good Spring threading with a TaskExecutor examples?

但是我需要做些什么才能维护两个线程之间的事务?我尝试过使用@Transactional注释run() - 方法之类的东西,但这并不起作用。

1 个答案:

答案 0 :(得分:6)

很少有人会想到这些问题 -

全球交易
您的所有线程是否应该属于同一个事务?这意味着,如果一个线程抛出异常,你想要回滚其他线程中的更改吗?

本地交易 每个线程是否应该拥有自己的事务边界而不影响其他线程?在这种情况下,一个线程中的异常和回滚不会影响其他线程。

要实现全局事务,您很可能必须使用具有适当资源驱动程序的JTA事务管理器。 Spring没有对全局事务的默认支持。如果您的应用程序在Java EE应用程序服务器(Jboss,Weblogic)中运行,则默认情况下您将获得JTA支持,您可以配置spring以使用应用程序服务器的事务管理功能。 如果您在tomcat中运行,则可以插入一个开源JTA实现,如 -

Atomikos

Jboss Transactions

为了实现本地事务(即每个线程都有自己的事务边界),我认为除了使用spring transactional annotations之外,你不会做任何事情。 在下面的示例中,只需确保'service'是一个spring bean并且doSomething()被适当地注释。

taskExecutor.execute( new Runnable() {
           public void run() {
                service.doSomething( );
           }
      });