ix.IxGroupBy Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of ixjava Show documentation
Show all versions of ixjava Show documentation
ixjava developed by David Karnok
/*
* Copyright 2011-2016 David Karnok
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package ix;
import java.util.*;
final class IxGroupBy extends IxSource> {
static final Object NULL = new Object();
final IxFunction super T, ? extends K> keySelector;
final IxFunction super T, ? extends V> valueSelector;
IxGroupBy(Iterable source, IxFunction super T, ? extends K> keySelector,
IxFunction super T, ? extends V> valueSelector) {
super(source);
this.keySelector = keySelector;
this.valueSelector = valueSelector;
}
@Override
public Iterator> iterator() {
return new GroupByIterator(source.iterator(), keySelector, valueSelector);
}
static final class GroupByIterator extends IxSourceIterator> {
final IxFunction super T, ? extends K> keySelector;
final IxFunction super T, ? extends V> valueSelector;
final Map> groups;
final Queue> queue;
GroupByIterator(Iterator it, IxFunction super T, ? extends K> keySelector,
IxFunction super T, ? extends V> valueSelector) {
super(it);
this.keySelector = keySelector;
this.valueSelector = valueSelector;
this.groups = new HashMap>();
this.queue = new ArrayDeque>();
}
@Override
protected boolean moveNext() {
GroupedIterable g = queue.poll();
if (g != null) {
value = g;
hasValue = true;
return true;
}
if (mainMoveNext()) {
g = queue.poll();
value = g;
hasValue = true;
return true;
}
done = true;
return false;
}
boolean mainMoveNext() {
for (;;) {
if (!it.hasNext()) {
groups.clear();
return false;
}
T v = it.next();
K key = keySelector.apply(v);
V val = valueSelector.apply(v);
GroupedIterable g = groups.get(key);
if (g == null) {
g = new GroupedIterable(key, this);
groups.put(key, g);
g.iterator.queue.offer(val != null ? val : NULL);
queue.offer(g);
return true;
} else {
g.iterator.queue.offer(val != null ? val : NULL);
}
}
}
boolean groupMoveNext(GroupByGroupIterator groupIterator) {
if (done) {
return false;
}
for (;;) {
if (!it.hasNext()) {
groups.clear();
return false;
}
T v = it.next();
K key = keySelector.apply(v);
V val = valueSelector.apply(v);
GroupedIterable g = groups.get(key);
if (g == null) {
g = new GroupedIterable(key, this);
groups.put(key, g);
g.iterator.queue.offer(val != null ? val : NULL);
queue.offer(g);
} else {
g.iterator.queue.offer(val != null ? val : NULL);
if (EqualityHelper.INSTANCE.test(key, groupIterator.key)) {
return true;
}
}
}
}
}
static final class GroupedIterable extends GroupedIx {
final GroupByGroupIterator iterator;
boolean once;
GroupedIterable(K key, GroupByIterator, K, V> parent) {
super(key);
this.iterator = new GroupByGroupIterator(parent, key);
}
@Override
public Iterator iterator() {
if (!once) {
once = true;
return iterator;
}
throw new IllegalStateException("This GroupedIx iterable can be consumed only once.");
}
@Override
public String toString() {
return "GroupedIterable[key=" + iterator.key + ", queue=" + iterator.queue.size() + "]";
}
}
static final class GroupByGroupIterator extends IxBaseIterator {
final GroupByIterator, K, V> parent;
final K key;
final ArrayDeque
© 2015 - 2025 Weber Informatics LLC | Privacy Policy