Poniższy kod działa, ale Hibernate nigdy nie przestaje chwytać żadnego obiektu. Wywołanie session.clear()
powoduje, że wyjątki dotyczące pobierania połączonej klasy i wywoływania session.evict(currentObject)
przed pobraniem następnego obiektu również nie zwalnia pamięci. W końcu wyczerpuję przestrzeń sterty.Hibernate: spaceruj po milionach wierszy i nie przeciekaj pamięci
Sprawdzając moje zrzuty sterty, StatefulPersistenceContext jest korzeniem odśmiecania dla wszystkich odniesień wskazujących na moje obiekty.
public class CriteriaReportSource implements JRDataSource {
private ScrollableResults sr;
private Object currentObject;
private Criteria c;
private static final int scrollSize = 10;
private int offset = 1;
public CriteriaReportSource(Criteria c) {
this.c = c;
advanceScroll();
}
private void advanceScroll() {
// ((Session) Main.em.getDelegate()).clear();
this.sr = c.setFirstResult(offset)
.setMaxResults(scrollSize)
.scroll(ScrollMode.FORWARD_ONLY);
offset += scrollSize;
}
public boolean next() {
if (sr.next()) {
currentObject = sr.get(0);
if (sr.isLast()) {
advanceScroll();
}
return true;
}
return false;
}
public Object getFieldValue(JRField jrf) throws JRException {
Object retVal = null;
if(currentObject == null) { return null; }
try {
retVal = PropertyUtils.getProperty(currentObject, jrf.getName());
} catch (Exception ex) {
Logger.getLogger(CriteriaReportSource.class.getName()).log(Level.SEVERE, null, ex);
}
return retVal;
}
}
Widziałem przykłady tego w [odniesienie do Hibernacja] (http://docs.jboss.org/hibernate/core/3.3/reference/en/html/batch.html#batch-update), jednak istnieje 'session.flush()' zostało wywołane przed 'session.clear()'. Czy możesz spróbować, czy to robi różnicę? –
Jaka jest baza danych? Nie wszystkie obsługują prawdziwe przewijanie kursora. Ponadto nie widzę, abyś zamknął ScrollableResults. –
Używam MySQL. –