11

我打算为 EJB 编写一个拦截器,它基本上会执行以下操作:

@AroundInvoke
public Object setContext(InvocationContext ctx) throws Exception {
    em.createQuery("... set something [database specific] ...").getSingleResult();
    try {
        return ctx.proceed();
    } finally {
        em.flush();
        em.createQuery("... unset something [database specific] ...").getSingleResult();
    }
}

问题是如果将 em.flush() 应用于使用 @TransactionAttribute(NOT_SUPPORTED) 或 @TransactionAttribute(SUPPORTS) 注释的方法(即只读方法),则 em.flush() 会引发异常,而它适用于使用默认 @ 的方法TransactionAttribute(REQUIRED)(即更改数据库的方法)。

有没有办法判断事务是否处于活动状态,以避免在当前没有事务运行时调用 em.flush(),或者我必须创建两个拦截器,一个带有 em.flush(),一个不带它?

我无法注入 UserTransaction,因为 bean 使用容器管理的事务。EntityManager 的 (em) 事务方法仅适用于 resource_local 事务,不适用于 JTA 事务。我需要 em.flush() 调用以确保未设置的查询是运行的最后一件事。

作为奖励问题,有谁知道我想要做的是否是某种不好的做法?我知道拦截器在与 bean 相同的上下文中运行,因此通过数据库会话共享内容是安全的,但我必须非常小心地在退出时取消设置所有内容,因为连接被汇集在容器中。

4

1 回答 1

21

您可以注入一个TransactionSynchronizationRegistry并使用getTransactionStatus在当前上下文中获取事务的状态,它返回一个状态int类中的常量,在您正在寻找的情况下STATUS_NO_TRANSACTION

注入:

@Resource
TransactionSynchronizationRegistry tsr;

检查交易状态:

if (Status.STATUS_NO_TRANSACTION == tsr.getTransactionStatus()) {
   // no transaction
}
于 2012-12-22T19:25:05.267 回答