-
-
Notifications
You must be signed in to change notification settings - Fork 639
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Fixes memory leak of grouped iterator #1907
Changes from 5 commits
aa2c59b
c442771
cb7abb9
76b2c00
657175c
17b70a9
58413b6
5f84292
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,75 @@ | ||
/* / \____ _ _ ____ ______ / \ ____ __ _______ | ||
* / / \/ \ / \/ \ / /\__\/ // \/ \ // /\__\ JΛVΛSLΛNG | ||
* _/ / /\ \ \/ / /\ \\__\\ \ // /\ \ /\\/ \ /__\ \ Copyright 2014-2017 Javaslang, http://javaslang.io | ||
* /___/\_/ \_/\____/\_/ \_/\__\/__/\__\_/ \_// \__/\_____/ Licensed under the Apache License, Version 2.0 | ||
*/ | ||
package javaslang.collection; | ||
|
||
import javaslang.JmhRunner; | ||
import org.junit.Test; | ||
import org.openjdk.jmh.annotations.*; | ||
import org.openjdk.jmh.infra.Blackhole; | ||
|
||
import java.util.Random; | ||
|
||
import static java.util.Arrays.asList; | ||
import static javaslang.JmhRunner.Includes.*; | ||
import static javaslang.JmhRunner.getRandomValues; | ||
import static scala.collection.JavaConversions.asScalaBuffer; | ||
|
||
@SuppressWarnings({ "ALL", "unchecked", "rawtypes" }) | ||
public class IteratorBenchmark { | ||
|
||
static final Array<Class<?>> CLASSES = Array.of( | ||
Sliding.class | ||
); | ||
|
||
@Test | ||
public void testAsserts() { JmhRunner.runDebugWithAsserts(CLASSES); } | ||
|
||
public static void main(String... args) { | ||
JmhRunner.runDebugWithAsserts(CLASSES); | ||
JmhRunner.runNormalNoAsserts(CLASSES, JAVA, SCALA, JAVASLANG); | ||
} | ||
|
||
@State(Scope.Benchmark) | ||
public static class Base { | ||
@Param({ "10", "100", "1000" }) | ||
public int CONTAINER_SIZE; | ||
|
||
Integer[] ELEMENTS; | ||
|
||
/* Only use these for non-mutating operations */ | ||
java.util.List<Integer> javaMutable; | ||
scala.collection.Iterator<Integer> scalaPersistent; | ||
javaslang.collection.Iterator<Integer> slangPersistent; | ||
|
||
@Setup | ||
public void setup() { | ||
final Random random = new Random(0); | ||
ELEMENTS = getRandomValues(CONTAINER_SIZE, false, random); | ||
javaMutable = asList(ELEMENTS); | ||
scalaPersistent = (scala.collection.Iterator<Integer>) scala.collection.immutable.Seq$.MODULE$.apply(asScalaBuffer(javaMutable)).iterator(); | ||
slangPersistent = Iterator.of(ELEMENTS); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @paplorinc |
||
} | ||
} | ||
|
||
public static class Sliding extends Base { | ||
|
||
@Benchmark | ||
public void scala_persistent(Blackhole bh) { | ||
final scala.collection.Iterator.GroupedIterator values = scalaPersistent.sliding(3, 1); | ||
while (values.hasNext()) { | ||
bh.consume(values.next()); | ||
} | ||
} | ||
|
||
@Benchmark | ||
public void slang_persistent(Blackhole bh) { | ||
final Iterator<Seq<Integer>> values = slangPersistent.sliding(3); | ||
while (values.hasNext()) { | ||
bh.consume(values.next()); | ||
} | ||
} | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -11,6 +11,7 @@ | |
import javaslang.Tuple3; | ||
import javaslang.collection.IteratorModule.ConcatIterator; | ||
import javaslang.collection.IteratorModule.DistinctIterator; | ||
import javaslang.collection.IteratorModule.GroupedIterator; | ||
import javaslang.control.Option; | ||
|
||
import java.math.BigDecimal; | ||
|
@@ -20,6 +21,7 @@ | |
import static java.lang.Double.NEGATIVE_INFINITY; | ||
import static java.lang.Double.POSITIVE_INFINITY; | ||
import static java.math.RoundingMode.HALF_UP; | ||
import static javaslang.API.*; | ||
import static javaslang.collection.IteratorModule.BigDecimalHelper.areEqual; | ||
import static javaslang.collection.IteratorModule.BigDecimalHelper.asDecimal; | ||
import static javaslang.collection.IteratorModule.EmptyIterator; | ||
|
@@ -1535,9 +1537,9 @@ default <C> Map<C, Iterator<T>> groupBy(Function<? super T, ? extends C> classif | |
|
||
@Override | ||
default Iterator<Seq<T>> grouped(int size) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Scala's grouped() and sliding() functions return a It would be nice if we could provide the same functionality. Returning a There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Note: I internally based the buffer on our persistent Vector, Scala users a mutable ArrayBuffer. Our version is blazing fast (a little slower than the mutable version). Congrats @paplorinc !!! There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. nice, could you share some benchmarks :)? |
||
return sliding(size, size); | ||
return new GroupedIterator<>(this, size, size); | ||
} | ||
|
||
@Override | ||
default boolean hasDefiniteSize() { | ||
return false; | ||
|
@@ -1827,37 +1829,9 @@ default Iterator<Seq<T>> sliding(int size) { | |
|
||
@Override | ||
default Iterator<Seq<T>> sliding(int size, int step) { | ||
if (size <= 0 || step <= 0) { | ||
throw new IllegalArgumentException("size: " + size + " or step: " + step + " not positive"); | ||
} | ||
if (!hasNext()) { | ||
return empty(); | ||
} else { | ||
final Stream<T> source = Stream.ofAll(this); | ||
return new AbstractIterator<Seq<T>>() { | ||
private Stream<T> that = source; | ||
private IndexedSeq<T> next = null; | ||
|
||
@Override | ||
public boolean hasNext() { | ||
while (next == null && !that.isEmpty()) { | ||
final Tuple2<Stream<T>, Stream<T>> split = that.splitAt(size); | ||
next = split._1.toVector(); | ||
that = split._2.isEmpty() ? Stream.empty() : that.drop(step); | ||
} | ||
return next != null; | ||
} | ||
|
||
@Override | ||
public IndexedSeq<T> getNext() { | ||
final IndexedSeq<T> result = next; | ||
next = null; | ||
return result; | ||
} | ||
}; | ||
} | ||
return new GroupedIterator<>(this, size, step); | ||
} | ||
|
||
@Override | ||
default Tuple2<Iterator<T>, Iterator<T>> span(Predicate<? super T> predicate) { | ||
Objects.requireNonNull(predicate, "predicate is null"); | ||
|
@@ -2089,6 +2063,40 @@ public String toString() { | |
} | ||
} | ||
|
||
final class GroupedIterator<T> implements Iterator<Seq<T>> { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I don't know if we should really switch to Iterator (instead of AbstractIterator) in version 2.1.0 for slightly better performance... In 3.0.0 I think we should use the faster Iterator (without a pretty toString()). |
||
|
||
private final Iterator<T> that; | ||
private final int step; | ||
private final int gap; | ||
|
||
private List<T> buffer; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
|
||
|
||
GroupedIterator(Iterator<T> that, int size, int step) { | ||
if (size < 1 || step < 1) { | ||
throw new IllegalArgumentException("size (" + size + ") and step (" + step + ") must both be positive"); | ||
} | ||
this.that = that; | ||
this.step = step; | ||
this.gap = Math.max(step - size, 0); | ||
this.buffer = List.ofAll(that.take(size)); | ||
} | ||
|
||
@Override | ||
public boolean hasNext() { | ||
return !buffer.isEmpty(); | ||
} | ||
|
||
@Override | ||
public Seq<T> next() { | ||
if (!hasNext()) { | ||
throw new NoSuchElementException(); | ||
} | ||
final Seq<T> result = buffer; | ||
buffer = that.hasNext() ? that.drop(gap).take(step).toList().prependAll(buffer.drop(step)) : buffer.take(0); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I did appendAll() before with Vector. List.prependAll() performs better in this use-case. See above. I will add a benchmark. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
@Override
default List<T> appendAll(Iterable<? extends T> elements) {
Objects.requireNonNull(elements, "elements is null");
return List.<T> ofAll(elements).prependAll(this);
} There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I think about adding appendAll() to Traversable and therefor to Iterator. we could then operate entirely on the Iterator and finally call toList(). Maybe it is simpler to read. However, each operation requires a new Iterator instance. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Mmhh, does not seem to be applicable here... There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. If you're lucky it will end up on the stack (i.e. escape analysis), you should measure the impact :) There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I haven't done such measures yet - no clue which tool to use in order to see how things are organized within the JVM 🙄 do you have a hint? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
|
||
return result; | ||
} | ||
} | ||
|
||
final class BigDecimalHelper { | ||
|
||
@GwtIncompatible("Math::nextDown is not implemented") | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -720,6 +720,12 @@ default <U> List<T> distinctBy(Function<? super T, ? extends U> keyExtractor) { | |
|
||
@Override | ||
default List<T> drop(int n) { | ||
if (n <= 0) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 👍 |
||
return this; | ||
} | ||
if (n >= size()) { | ||
return empty(); | ||
} | ||
List<T> list = this; | ||
for (long i = n; i > 0 && !list.isEmpty(); i--) { | ||
list = list.tail(); | ||
|
@@ -1390,12 +1396,12 @@ default Option<List<T>> tailOption() { | |
|
||
@Override | ||
default List<T> take(int n) { | ||
if (n >= length()) { | ||
return this; | ||
} | ||
if (n <= 0) { | ||
return empty(); | ||
} | ||
if (n >= length()) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. what's the difference here? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. we do the length() check second now There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I see that, but why is it important? It's a constant There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. yes, I forgot, JIT magic There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. every runtime env? GWT resp Javascript? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. if I can sleep better we have gained s.th. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This could help you even more: https://rainymood.com There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 😴 There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I use it sometimes when the environment is noisy, it's very relaxing :D There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I will try it at $work :) |
||
return this; | ||
} | ||
List<T> result = Nil.instance(); | ||
List<T> list = this; | ||
for (int i = 0; i < n; i++, list = list.tail()) { | ||
|
@@ -1406,12 +1412,12 @@ default List<T> take(int n) { | |
|
||
@Override | ||
default List<T> takeRight(int n) { | ||
if (n >= length()) { | ||
return this; | ||
} | ||
if (n <= 0) { | ||
return empty(); | ||
} | ||
if (n >= length()) { | ||
return this; | ||
} | ||
return reverse().take(n).reverse(); | ||
} | ||
|
||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
isn't this a mutable datastructure?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
oh, forgot to remove the comment
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
but if it's mutable, can't we use a simple
ArrayList
in the background, instead of a linked list?Edit: are iterators supposed to be called from multi-threaded environments?