blob: 06accf918e39ac56c86ae0d61a9c7887946aeb96 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 1998, 2013 Oracle. All rights reserved.
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License v1.0 and Eclipse Distribution License v. 1.0
* which accompanies this distribution.
* The Eclipse Public License is available at http://www.eclipse.org/legal/epl-v10.html
* and the Eclipse Distribution License is available at
* http://www.eclipse.org/org/documents/edl-v10.php.
*
* Contributors:
* dclarke - initial
******************************************************************************/
package eclipselink.example.jpa.employee.services.paging;
import java.util.List;
import javax.persistence.EntityManager;
import javax.persistence.EntityManagerFactory;
import javax.persistence.TypedQuery;
import javax.persistence.criteria.CriteriaQuery;
import org.eclipse.persistence.config.HintValues;
import org.eclipse.persistence.config.QueryHints;
import org.eclipse.persistence.queries.CursoredStream;
import org.eclipse.persistence.queries.ScrollableCursor;
import eclipselink.example.jpa.employee.model.Employee;
/**
* Example of using an EclipseLink {@link CursoredStream} to page query results.
* Generally a stream is used to process query results in memory avoiding the
* cost of loading all results before starting. This example of using a stream
* for paging may not be
*
* @since EclipseLink 2.4.2
*/
public class StreamPaging<T> extends EntityPaging<T> {
private ScrollableCursor stream;
public StreamPaging(EntityManagerFactory emf, CriteriaQuery<Employee> criteria, int pageSize) {
super(null, pageSize);
EntityManager em = emf.createEntityManager();
try {
TypedQuery<?> query = em.createQuery(criteria);
query.setHint(QueryHints.SCROLLABLE_CURSOR, HintValues.TRUE);
this.stream = (ScrollableCursor) query.getSingleResult();
} finally {
em.close();
}
}
@Override
@SuppressWarnings("unchecked")
public List<T> next() {
if (!hasNext()) {
return super.next();
}
int quantity = getPageSize();
// Adjust size if there are less elements on stream then a full page
if ((quantity + this.stream.getPosition() - 1) > this.stream.size()) {
quantity = this.stream.size() - this.stream.getPosition();
}
List<T> entities = (List<T>) this.stream.next(quantity);
this.currentPage++;
return entities;
}
@Override
public List<T> get(int page) {
if (page == getCurrentPage() + 1) {
return next();
}
int position = (page - 1) * getPageSize();
this.stream.absolute(position);
this.currentPage = page - 1;
return next();
}
@Override
public int size() {
return this.stream.size();
}
@Override
public int getNumPages() {
return (size() / getPageSize()) + (size() % getPageSize() > 0 ? 1 : 0);
}
/**
* Release the stream and its corresponding resources.
*/
public void close() {
if (this.stream != null) {
this.stream.close();
this.stream = null;
}
}
}