Spring + Hibernate + Envers +多线程 - 会话关闭

Paw*_*żyk 7 java spring hibernate jpa hibernate-envers

我们使用Hibernate(使用JPA)和Hibernate Envers来持久化对象的历史.Web应用程序运行许多线程,其中一些是通过其他应用程序的RMI方法调用创建的,其中一些是由应用程序本身创建的,其中一些是为处理http请求而创建的(它们生成视图).

我们还使用Open Session In View模式来管理会话,因此我们的web.xml包含:

<filter>
    <filter-name>openEntityManagerInViewFilter</filter-name>
    <filter-class>org.springframework.orm.jpa.support.OpenEntityManagerInViewFilter</filter-class>
</filter>

<filter-mapping>
    <filter-name>openEntityManagerInViewFilter</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>
Run Code Online (Sandbox Code Playgroud)

使用DAO访问数据库,所有这些都有Spring注入的EntityManagers.

@PersistenceContext
protected EntityManager em;

@PersistenceUnit
protected EntityManagerFactory emf;
Run Code Online (Sandbox Code Playgroud)

在我们决定使用Hibernate Envers之前,一切都运行良好.当任何不是视图生成线程的线程运行代码以获取旧版本的对象时,抛出异常.

@Override
public O loadByRevision(Long revision, Long id) {
    @SuppressWarnings("unchecked")
    O object = (O) AuditReaderFactory.get(em).createQuery().forEntitiesAtRevision(getBaseClass(), revision.intValue())
            .add(AuditEntity.id().eq(id)).getSingleResult();
    return object;
}
Run Code Online (Sandbox Code Playgroud)

线程"Scheduler"中的异常org.hibernate.SessionException:会话已关闭!在Org.hibernate.envers.tools.quols.QueryBuilder.toQuery(QueryBuilder)的org.hibernate.internal.AbstractSessionImpl.errorIfClosed(AbstractSessionImpl.java:129)org.hibernate.internal.SessionImpl.createQuery(SessionImpl.java:1776) .java:226)org.hibernate.envers.query.impl.AbstractAuditQuery.buildQuery(AbstractAuditQuery.java:92)org.hibernate.envers.query.impl.EntitiesAtRevisionQuery.list(EntitiesAtRevisionQuery.java:108)at org. hibernate.envers.query.impl.AbstractAuditQuery.getSingleResult(AbstractAuditQuery.java:110)(..)

当上面的代码由视图生成线程运行时,它工作正常.此外,DAO中的非envers代码适用于每个线程.例如,下面的代码段

@Override
public O load(Long id) {
    final O find = em.find(getBaseClass(), id);
    return find;
}
Run Code Online (Sandbox Code Playgroud)

可以通过RMI线程运行而不会出现问题.

为什么非视图线程可以在没有异常的情况下调用实体管理器上的方法,但是不能将Envers的AuditReaderFactory与该实体管理器一起使用?我认为可能在实体管理器上调用方法会创建一个临时会话但是在使用Envers时不会发生,这是真的吗?

解决该问题的最佳方法是什么(以便可以从每个线程使用AuditReaderFactory)?

Paw*_*żyk 3

我们没有找到为什么在非 ui 线程中方法调用EntityManagerFactory有效但方法调用AuditReaderFactory无效。无论如何,我们找到了解决问题的方法。

解决方案是用 注释方法@Transactional。如果在调用 AuditReaderFactory 之前调用链中的任何方法被标记为,则非 ui 线程中@Transactional没有该方法。SessionException

事实证明,仅仅实现loadByRevision事务性还不够。如果该方法返回的对象包含延迟加载的持久包,则会导致在loadByRevision方法范围之外访问它们LazyInitializationException(没有会话)。

最终的解决方案是确保如果任何线程想要从数据库加载一些数据,所有加载(获取对象和访问延迟加载的集合)都将在一个用@Transactional.