微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

加载Java实体时内存不足

我有一个可怕的问题,希望有一个非常简单的答案.当我执行基本操作时,我的内存不足

如果我有这样的代码

MyEntity myEntity;
for (Object id: someIdList) {
   myEntity = find(id); 
   // do something basic with myEntity
}

而find()方法一个标准的EntityManager相关方法

public MyEntity find(Object id) {
    return em.find(mycorp.ejb.entity.MyEntity.class,id);
}

这段代码工作了几周,如果数据库中的项目较少,则可以正常工作.我面临的结果是:

java.lang.OutOfMemoryError:超出了GC开销限制

异常来自oracle toplink调用一些oracle jdbc方法.

循环存在是因为EJBQL(例如“将MyEntity中的选择对象(o)作为o”)会在有大量记录时使应用程序服务器过载.

最佳答案

This code worked a couple of weeks ago,and works fine if there are fewer items in the database. The resulting error I am facing is: java.lang.OutOfMemoryError: GC overhead limit exceeded

这里没什么好奇怪的.由em.find()加载的实体被放置并保存在持久化上下文(在内存中)以跟踪更改,因此如果您在没有预防措施的情况下批量加载过多的实体,那么您将只是爆炸内存并获得OOME.

如果您确实需要对所有实体执行某些操作,则需要先调用flush()将所有更改推送到数据库,然后清除()以清除持久上下文并定期释放内存:

int i = 0;
for (Object id: someReallyBigIdList) {
    myEntity = find(id); 
    // do something basic with myEntity
    if ( i % 20 == 0 ) { //20,same as the JDBC batch size
        //flush a batch of DML operations and release memory:
        em.flush();
        em.clear();
    }
    i++;
}

调用clear()会导致所有托管实体分离.对未刷新到数据库的实体所做的更改将不会保留.因此需要首先刷新()更改.

原文地址:https://www.jb51.cc/java/438187.html

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐