Entitymanager导致内存泄漏?

GBa*_*GBa 9 java myeclipse jpa

我的Java应用程序中的内存泄漏很慢.我想知道这是否可能是由于在使用时并不总是关闭Entitymanager.但是使用myeclipse生成数据库代码,我得到的方法如下:

public Meit update(Meit entity) {
    logger.info("updating Meit instance");
    try {
        Meit result = getEntityManager().merge(entity);
        logger.info("update successful");
        return result;
    } catch (RuntimeException re) {
        logger.error("update failed");
        throw re;
    }
}
Run Code Online (Sandbox Code Playgroud)

从不关闭EntityManager.考虑到这是生成的代码,我想知道谁是对的,我还是IDE.

ece*_*ulm 6

正如@Ruggs所说,如果您自己管理EntityManager生命周期(而不是由J2EE完成CMP容器管理持久性),那么您需要自己关闭EntityManager或至少调用EntityManager.clear()分离实体.

EntityManager是轻量级对象,因此不需要只有一个,您可以为每个事务创建一个并在提交事务后关闭它.

通过EntityManager加载/持久存储的所有实体都会保留在内存中,直到您明确地从中分离实体(通过EntityManager.detach()EntityManager.clear()EntityManager.close()).因此,拥有短期的EntityManagers会更好.如果您在获得OOME之后通过相同的EntityManager持久保存1000000个实体而不分离它们(如果您将每个实体保留在其自己的EntityTransaction中则无关紧要).

这篇文章都在http://javanotepad.blogspot.com/2007/06/how-to-close-jpa-entitymanger-in-web.html中进行了解释.

作为一个例子(取自前面的帖子),如果你想避免"内存泄漏",你应该做这样的事情(如果你没有使用CMP):

EntityManager em = emf.createEntityManager();

try {
  EntityTransaction t = em.getTransaction();
  try {
    t.begin();  
    // business logic to update the customer
    em.merge(cust);
    t.commit();
  } finally {
   if (t.isActive()) t.rollback();
  }
} finally {
  em.close();
}
Run Code Online (Sandbox Code Playgroud)


cle*_*tus 1

实体管理器通常应具有与应用程序相同的生命周期,并且不应根据每个请求创建或销毁。

您的“内存泄漏”可能只不过是缓存 JPA 所做的事情。您没有说明您使用哪个 JPA 提供程序,但我从经验中知道 EclipseLink 默认情况下会进行广泛的缓存(这是 JPA 和 ORM 所谓的总体优势的一部分)。

你怎么知道你有内存泄漏?