diff --git a/janusgraph-backend-testutils/src/main/java/org/janusgraph/graphdb/JanusGraphIndexTest.java b/janusgraph-backend-testutils/src/main/java/org/janusgraph/graphdb/JanusGraphIndexTest.java index 0969d06177..faa7db4e64 100644 --- a/janusgraph-backend-testutils/src/main/java/org/janusgraph/graphdb/JanusGraphIndexTest.java +++ b/janusgraph-backend-testutils/src/main/java/org/janusgraph/graphdb/JanusGraphIndexTest.java @@ -1228,6 +1228,7 @@ public void testGraphCentricQueryProfiling() { // satisfied by two graph-centric queries, one is mixed index query and the other one is full scan newTx(); assertEquals(3, tx.traversal().V().or(__.has("name", "bob"), __.has("age", 20)).count().next()); + assertEquals(3, tx.traversal().V().or(__.has("name", "bob"), __.has("age", 20)).order().by("age").count().next()); assertEquals(3, tx.traversal().V().or(__.has("name", "bob"), __.has("age", 20)).toList().size()); Metrics mMixedOr = tx.traversal().V().or(__.has("name", "bob"), __.has("age", 20)) .profile().next().getMetrics(0); diff --git a/janusgraph-core/src/main/java/org/janusgraph/graphdb/util/MultiDistinctOrderedIterator.java b/janusgraph-core/src/main/java/org/janusgraph/graphdb/util/MultiDistinctOrderedIterator.java index ec60ddd4bf..ff6990e3e8 100644 --- a/janusgraph-core/src/main/java/org/janusgraph/graphdb/util/MultiDistinctOrderedIterator.java +++ b/janusgraph-core/src/main/java/org/janusgraph/graphdb/util/MultiDistinctOrderedIterator.java @@ -23,33 +23,37 @@ import java.util.ArrayList; import java.util.Comparator; +import java.util.HashMap; import java.util.HashSet; import java.util.Iterator; import java.util.LinkedHashMap; import java.util.List; import java.util.Map; +import java.util.PriorityQueue; import java.util.Set; -import java.util.TreeMap; +public class MultiDistinctOrderedIterator extends CloseableAbstractIterator { -public class MultiDistinctOrderedIterator implements CloseableIterator { - - private final Map> iterators = new LinkedHashMap<>(); - private final Map values = new LinkedHashMap<>(); - private final TreeMap currentElements; + private final Map> iterators; + private final PriorityQueue queue; private final Set allElements = new HashSet<>(); + private final List iteratorValues; + private final Map iteratorIdx; private final Integer limit; private long count = 0; public MultiDistinctOrderedIterator(final Integer lowLimit, final Integer highLimit, final List> iterators, final List orders) { this.limit = highLimit; - final List> comp = new ArrayList<>(); - orders.forEach(o -> comp.add(new ElementValueComparator(o.key, o.order))); - Comparator comparator = new MultiComparator<>(comp); + this.iterators = new LinkedHashMap<>(iterators.size()); + this.iteratorValues = new ArrayList<>(iterators.size()); + this.iteratorIdx = new HashMap<>(iterators.size()); for (int i = 0; i < iterators.size(); i++) { this.iterators.put(i, iterators.get(i)); + this.iteratorValues.add(null); } - currentElements = new TreeMap<>(comparator); + final List> comp = new ArrayList<>(); + orders.forEach(o -> comp.add(new ElementValueComparator(o.key, o.order))); + this.queue = new PriorityQueue<>(iterators.size(), new MultiComparator<>(comp)); long i = 0; while (i < lowLimit && this.hasNext()) { this.next(); @@ -58,12 +62,12 @@ public MultiDistinctOrderedIterator(final Integer lowLimit, final Integer highLi } @Override - public boolean hasNext() { + protected E computeNext() { if (limit != null && limit != Query.NO_LIMIT && count >= limit) { - return false; + return endOfData(); } for (int i = 0; i < iterators.size(); i++) { - if (!values.containsKey(i) && iterators.get(i).hasNext()){ + if (iteratorValues.get(i) == null && iterators.get(i).hasNext()) { E element = null; do { element = iterators.get(i).next(); @@ -72,24 +76,25 @@ public boolean hasNext() { } } while (element == null && iterators.get(i).hasNext()); if (element != null) { - values.put(i, element); - currentElements.put(element, i); + iteratorValues.set(i, element); + iteratorIdx.put(element, i); + queue.add(element); allElements.add(element.id()); } } } - return !values.isEmpty(); - } - - @Override - public E next() { - count++; - return values.remove(currentElements.remove(currentElements.firstKey())); + if (queue.isEmpty()) { + return endOfData(); + } else { + count++; + E result = queue.remove(); + iteratorValues.set(iteratorIdx.remove(result), null); + return result; + } } @Override public void close() { iterators.values().forEach(CloseableIterator::closeIterator); } - }