一尘不染

调用EntityManager.getTransaction()时出现EJBException

hibernate

这可能有些琐碎,但我希望获得一些帮助。

我得到:

 javax.ejb.EJBException: java.lang.IllegalStateException: Illegal to call this method from injected, managed EntityManager 
 11:54:37,105 ERROR [STDERR] at org.jboss.ejb3.tx.Ejb3TxPolicy.handleExceptionInOurTx(Ejb3TxPolicy.java:77)
 11:54:37,105 ERROR [STDERR] at org.jboss.aspects.tx.TxPolicy.invokeInOurTx(TxPolicy.java:83)
 11:54:37,105 ERROR [STDERR] at org.jboss.aspects.tx.TxInterceptor$Required.invoke(TxInterceptor.java:190)

做的时候:

@PersistenceContext(unitName = "someName")
private EntityManager em;

...

final EntityManager entityManager = getEntityManager();
final EntityTransaction tx = entityManager.getTransaction(); // here

谁能告诉我原因是什么?


阅读 249

收藏
2020-06-20

共1个答案

一尘不染

在Java
EE受管上下文中,获取对与EntityManager关联的EntityTransaction实例的引用是非法的。从EntityManager.getTransaction()的Java
EE API文档中:

返回资源级的EntityTransaction对象。EntityTransaction实例可以顺序使用以开始和提交多个事务。

Returns:
    EntityTransaction instance
Throws:
    IllegalStateException - if invoked on a JTA entity manager

最后一行与此相关。

当使用@PersistenceContext或@Inject批注将EntityManager注入部署在应用程序服务器上的EJB中时,EntityManager将由容器而不是由应用程序管理。容器管理的实体管理器必须是JTA实体管理器;应用程序管理的实体管理器可以是资源本地实体管理器。这由JPA规范规定:

通过JTA控制基础交易的实体管理器称为JTA实体管理器。

由应用程序通过EntityTransaction API控制其基础事务的实体管理器称为资源本地实体管理器。

容器管理的实体管理器必须是JTA实体管理器。 JTA实体管理器仅指定用于Java EE容器。

从第一点推断(关于IllegalStateException),您必须不获取容器注入的EntityManagers的EntityTransaction参考。但是,如果容器仅注入EntityManagerFactory,并且您的应用程序通过调用来获得EntityManager引用,则可以这样做EntityManagerFactory.getEntityManager

另外,应该注意,EntityManager.getTransaction()对于JTA实体管理者而言,调用是没有意义的。JPA规范在EntityTransaction接口的定义中指出了这一点:

EntityTransaction接口用于控制资源本地实体管理器上的资源事务。

关于管理JTA事务本身的主题,如果您需要自己管理事务边界(即使用Bean管理的事务),请注入UserTransaction实例。或者,如果您希望容器管理事务,则只需使用适当的TransactionalAttribute值注释方法或bean

在应用程序服务器中,将本地资源实体管理器(和数据源)与bean管理的事务或容器管理的事务一起使用通常不是一个好主意,但这是可以做到的。

您可以在Hibernate
EntityManager文档中
找到一个合适的示例来演示BMT配合使用EntityManager的使用。如果您已经注释了bean类或方法,则CMT更为简单。您只需避免调用getEntityTransaction()CMT起作用的方法。

如果您希望进一步了解,我建议阅读JPA 2.0规范的第7章,标题为“实体管理器和持久性上下文”。本章中提供的示例演示:

  • 如何在应用服务器(通常是使用它们的地方)中使用JTA实体管理器。
  • 如何在应用程序服务器中使用资源本地实体管理器。
  • 如何在Java SE应用程序中使用资源本地实体管理器。
2020-06-20