JPA: what is the proper pattern for iterating over large result sets?

Page 537 of Java Persistence with Hibernate gives a solution using ScrollableResults, but alas it’s only for Hibernate.

So it seems that using setFirstResult/setMaxResults and manual iteration really is necessary. Here’s my solution using JPA:

private List<Model> getAllModelsIterable(int offset, int max)
{
    return entityManager.createQuery("from Model m", Model.class).setFirstResult(offset).setMaxResults(max).getResultList();
}

then, use it like this:

private void iterateAll()
{
    int offset = 0;

    List<Model> models;
    while ((models = Model.getAllModelsIterable(offset, 100)).size() > 0)
    {
        entityManager.getTransaction().begin();
        for (Model model : models)
        {
            log.info("do something with model: " + model.getId());
        }

        entityManager.flush();
        entityManager.clear();
        em.getTransaction().commit();
        offset += models.size();
    }
}

Leave a Comment