org.eclipse.collections.impl.bag.immutable.ImmutableArrayBag Maven / Gradle / Ivy
/*
* Copyright (c) 2016 Goldman Sachs.
* 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 accompany 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.
*/
package org.eclipse.collections.impl.bag.immutable;
import java.io.Serializable;
import java.util.Comparator;
import java.util.Iterator;
import java.util.NoSuchElementException;
import java.util.Optional;
import org.eclipse.collections.api.bag.Bag;
import org.eclipse.collections.api.bag.ImmutableBag;
import org.eclipse.collections.api.bag.MutableBag;
import org.eclipse.collections.api.block.function.Function;
import org.eclipse.collections.api.block.predicate.Predicate;
import org.eclipse.collections.api.block.predicate.Predicate2;
import org.eclipse.collections.api.block.predicate.primitive.IntPredicate;
import org.eclipse.collections.api.block.procedure.Procedure;
import org.eclipse.collections.api.block.procedure.primitive.ObjectIntProcedure;
import org.eclipse.collections.api.map.MutableMap;
import org.eclipse.collections.api.multimap.bag.ImmutableBagMultimap;
import org.eclipse.collections.api.ordered.OrderedIterable;
import org.eclipse.collections.api.set.ImmutableSet;
import org.eclipse.collections.api.tuple.Pair;
import org.eclipse.collections.impl.bag.mutable.HashBag;
import org.eclipse.collections.impl.block.factory.Predicates2;
import org.eclipse.collections.impl.factory.Bags;
import org.eclipse.collections.impl.map.mutable.UnifiedMap;
import org.eclipse.collections.impl.multimap.bag.HashBagMultimap;
import org.eclipse.collections.impl.set.mutable.UnifiedSet;
import org.eclipse.collections.impl.utility.ArrayIterate;
import org.eclipse.collections.impl.utility.Iterate;
/**
* @since 1.0
*/
public class ImmutableArrayBag
extends AbstractImmutableBag
implements Serializable
{
static final int MAXIMUM_USEFUL_ARRAY_BAG_SIZE = 20;
private static final long serialVersionUID = 1L;
private final T[] keys;
private final int[] counts;
ImmutableArrayBag(T[] keys, int[] counts)
{
this.keys = keys;
this.counts = counts;
if (this.keys.length != this.counts.length)
{
throw new IllegalArgumentException();
}
}
public static ImmutableArrayBag newBagWith(T... elements)
{
return ImmutableArrayBag.copyFrom(Bags.mutable.with(elements));
}
public static ImmutableArrayBag copyFrom(Bag bag)
{
int distinctItemCount = bag.sizeDistinct();
T[] newKeys = (T[]) new Object[distinctItemCount];
int[] newCounts = new int[distinctItemCount];
bag.forEachWithOccurrences(new ObjectIntProcedure()
{
private int index;
public void value(T each, int count)
{
newKeys[this.index] = each;
newCounts[this.index] = count;
this.index++;
}
});
return new ImmutableArrayBag<>(newKeys, newCounts);
}
@Override
public void forEachWithOccurrences(ObjectIntProcedure super T> objectIntProcedure)
{
for (int i = 0; i < this.keys.length; i++)
{
objectIntProcedure.value(this.keys[i], this.counts[i]);
}
}
@Override
public int sizeDistinct()
{
return this.keys.length;
}
@Override
public int size()
{
int sum = 0;
for (int value : this.counts)
{
sum += value;
}
return sum;
}
@Override
public int occurrencesOf(Object item)
{
int index = ArrayIterate.detectIndexWith(this.keys, Predicates2.equal(), item);
if (index > -1)
{
return this.counts[index];
}
return 0;
}
@Override
public ImmutableBag newWith(T element)
{
int elementIndex = ArrayIterate.detectIndexWith(this.keys, Predicates2.equal(), element);
int distinctItemCount = this.sizeDistinct() + (elementIndex == -1 ? 1 : 0);
if (distinctItemCount > MAXIMUM_USEFUL_ARRAY_BAG_SIZE)
{
return HashBag.newBag(this).with(element).toImmutable();
}
return this.newArrayBagWith(element, elementIndex, distinctItemCount);
}
private ImmutableBag newArrayBagWith(T element, int elementIndex, int distinctItemCount)
{
T[] newKeys = (T[]) new Object[distinctItemCount];
int[] newCounts = new int[distinctItemCount];
System.arraycopy(this.keys, 0, newKeys, 0, this.keys.length);
System.arraycopy(this.counts, 0, newCounts, 0, this.counts.length);
if (elementIndex == -1)
{
newKeys[distinctItemCount - 1] = element;
newCounts[distinctItemCount - 1] = 1;
}
else
{
newCounts[elementIndex]++;
}
return new ImmutableArrayBag<>(newKeys, newCounts);
}
@Override
public ImmutableBag newWithout(T element)
{
int elementIndex = ArrayIterate.detectIndexWith(this.keys, Predicates2.equal(), element);
if (elementIndex > -1)
{
int distinctItemCount = this.sizeDistinct() - (this.counts[elementIndex] == 1 ? 1 : 0);
T[] newKeys = (T[]) new Object[distinctItemCount];
int[] newCounts = new int[distinctItemCount];
if (distinctItemCount == this.sizeDistinct())
{
System.arraycopy(this.keys, 0, newKeys, 0, distinctItemCount);
System.arraycopy(this.counts, 0, newCounts, 0, distinctItemCount);
newCounts[elementIndex]--;
}
else
{
System.arraycopy(this.keys, 0, newKeys, 0, elementIndex);
System.arraycopy(this.counts, 0, newCounts, 0, elementIndex);
System.arraycopy(this.keys, elementIndex + 1, newKeys, elementIndex, newKeys.length - elementIndex);
System.arraycopy(this.counts, elementIndex + 1, newCounts, elementIndex, newCounts.length - elementIndex);
}
return new ImmutableArrayBag<>(newKeys, newCounts);
}
return this;
}
@Override
public MutableMap toMapOfItemToCount()
{
MutableMap map = UnifiedMap.newMap(this.size());
this.forEachWithOccurrences(map::put);
return map;
}
@Override
public ImmutableBag newWithAll(Iterable extends T> elements)
{
return Bags.immutable.withAll(Iterate.addAllTo(elements, HashBag.newBag(this)));
}
@Override
public ImmutableBag selectByOccurrences(IntPredicate predicate)
{
MutableBag result = HashBag.newBag();
this.forEachWithOccurrences((each, occurrences) -> {
if (predicate.accept(occurrences))
{
result.addOccurrences(each, occurrences);
}
});
return result.toImmutable();
}
@Override
public ImmutableBag selectInstancesOf(Class clazz)
{
MutableBag result = HashBag.newBag();
this.forEachWithOccurrences((each, index) -> {
if (clazz.isInstance(each))
{
result.addOccurrences((S) each, index);
}
});
return ImmutableArrayBag.copyFrom(result);
}
@Override
public ImmutableBagMultimap groupBy(Function super T, ? extends V> function)
{
return this.groupBy(function, HashBagMultimap.newMultimap()).toImmutable();
}
@Override
public ImmutableBagMultimap groupByEach(Function super T, ? extends Iterable> function)
{
return this.groupByEach(function, HashBagMultimap.newMultimap()).toImmutable();
}
@Override
public T getFirst()
{
return ArrayIterate.getFirst(this.keys);
}
@Override
public T getLast()
{
return ArrayIterate.getLast(this.keys);
}
@Override
public T getOnly()
{
if (this.counts.length == 0)
{
throw new IllegalStateException("Size must be 1 but was 0");
}
if (this.counts.length > 1 || this.counts[0] > 1)
{
throw new IllegalStateException("Size must be 1 but was greater than 1");
}
return this.getFirst();
}
@Override
public ImmutableBag select(Predicate super T> predicate)
{
return this.select(predicate, HashBag.newBag()).toImmutable();
}
@Override
public ImmutableBag reject(Predicate super T> predicate)
{
return this.reject(predicate, HashBag.newBag()).toImmutable();
}
@Override
public ImmutableBag collect(Function super T, ? extends V> function)
{
MutableBag result = this.collect(function, HashBag.newBag());
return ImmutableArrayBag.copyFrom(result);
}
@Override
public ImmutableBag collectIf(
Predicate super T> predicate,
Function super T, ? extends V> function)
{
MutableBag result = this.collectIf(predicate, function, HashBag.newBag());
return ImmutableArrayBag.copyFrom(result);
}
@Override
public ImmutableBag flatCollect(Function super T, ? extends Iterable> function)
{
return this.flatCollect(function, HashBag.newBag()).toImmutable();
}
@Override
public boolean equals(Object other)
{
if (this == other)
{
return true;
}
if (!(other instanceof Bag))
{
return false;
}
Bag> bag = (Bag>) other;
if (this.size() != bag.size())
{
return false;
}
for (int i = 0; i < this.keys.length; i++)
{
if (this.counts[i] != bag.occurrencesOf(this.keys[i]))
{
return false;
}
}
return true;
}
@Override
public int hashCode()
{
int sum = 0;
for (int i = 0; i < this.keys.length; i++)
{
T each = this.keys[i];
sum += (each == null ? 0 : each.hashCode()) ^ this.counts[i];
}
return sum;
}
@Override
public void each(Procedure super T> procedure)
{
for (int i = 0; i < this.keys.length; i++)
{
T key = this.keys[i];
for (int j = 1; j <= this.counts[i]; j++)
{
procedure.value(key);
}
}
}
@Override
public Iterator iterator()
{
return new ArrayBagIterator();
}
@Override
public boolean anySatisfy(Predicate super T> predicate)
{
return ArrayIterate.anySatisfy(this.keys, predicate);
}
@Override
public boolean anySatisfyWith(Predicate2 super T, ? super P> predicate, P parameter)
{
return ArrayIterate.anySatisfyWith(this.keys, predicate, parameter);
}
@Override
public boolean allSatisfy(Predicate super T> predicate)
{
return ArrayIterate.allSatisfy(this.keys, predicate);
}
@Override
public
boolean allSatisfyWith(Predicate2 super T, ? super P> predicate, P parameter)
{
return ArrayIterate.allSatisfyWith(this.keys, predicate, parameter);
}
@Override
public boolean noneSatisfy(Predicate super T> predicate)
{
return ArrayIterate.noneSatisfy(this.keys, predicate);
}
@Override
public
boolean noneSatisfyWith(Predicate2 super T, ? super P> predicate, P parameter)
{
return ArrayIterate.noneSatisfyWith(this.keys, predicate, parameter);
}
@Override
public T detect(Predicate super T> predicate)
{
return ArrayIterate.detect(this.keys, predicate);
}
@Override
public
T detectWith(Predicate2 super T, ? super P> predicate, P parameter)
{
return ArrayIterate.detectWith(this.keys, predicate, parameter);
}
@Override
public Optional detectOptional(Predicate super T> predicate)
{
return ArrayIterate.detectOptional(this.keys, predicate);
}
@Override
public Optional detectWithOptional(Predicate2 super T, ? super P> predicate, P parameter)
{
return ArrayIterate.detectWithOptional(this.keys, predicate, parameter);
}
@Override
public T min(Comparator super T> comparator)
{
return ArrayIterate.min(this.keys, comparator);
}
@Override
public T max(Comparator super T> comparator)
{
return ArrayIterate.max(this.keys, comparator);
}
@Override
public T min()
{
return ArrayIterate.min(this.keys);
}
@Override
public T max()
{
return ArrayIterate.max(this.keys);
}
@Override
public > T minBy(Function super T, ? extends V> function)
{
return ArrayIterate.minBy(this.keys, function);
}
@Override
public > T maxBy(Function super T, ? extends V> function)
{
return ArrayIterate.maxBy(this.keys, function);
}
/**
* @deprecated in 6.0. Use {@link OrderedIterable#zip(Iterable)} instead.
*/
@Override
@Deprecated
public ImmutableBag> zip(Iterable that)
{
return this.zip(that, HashBag.newBag()).toImmutable();
}
/**
* @deprecated in 6.0. Use {@link OrderedIterable#zipWithIndex()} instead.
*/
@Override
@Deprecated
public ImmutableSet> zipWithIndex()
{
return this.zipWithIndex(UnifiedSet.newSet()).toImmutable();
}
protected Object writeReplace()
{
return new ImmutableBagSerializationProxy<>(this);
}
private final class ArrayBagIterator
implements Iterator
{
private int position;
private int remainingOccurrences = -1;
private ArrayBagIterator()
{
this.remainingOccurrences = ImmutableArrayBag.this.sizeDistinct() > 0 ? ImmutableArrayBag.this.counts[0] : 0;
}
@Override
public boolean hasNext()
{
return this.position != ImmutableArrayBag.this.keys.length
&& !(this.position == ImmutableArrayBag.this.keys.length - 1 && this.remainingOccurrences == 0);
}
@Override
public T next()
{
if (!this.hasNext())
{
throw new NoSuchElementException();
}
T result = ImmutableArrayBag.this.keys[this.position];
this.remainingOccurrences--;
if (this.remainingOccurrences == 0)
{
this.position++;
if (this.position != ImmutableArrayBag.this.keys.length)
{
this.remainingOccurrences = ImmutableArrayBag.this.counts[this.position];
}
}
return result;
}
@Override
public void remove()
{
throw new UnsupportedOperationException("Cannot remove from an ImmutableArrayBag");
}
}
}