com.gs.collections.impl.bag.immutable.ImmutableArrayBag Maven / Gradle / Ivy
Show all versions of gs-collections Show documentation
/*
* Copyright 2014 Goldman Sachs.
*
* 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 com.gs.collections.impl.bag.immutable;
import java.io.Serializable;
import java.util.Collection;
import java.util.Comparator;
import java.util.Iterator;
import java.util.NoSuchElementException;
import com.gs.collections.api.bag.Bag;
import com.gs.collections.api.bag.ImmutableBag;
import com.gs.collections.api.bag.MutableBag;
import com.gs.collections.api.block.function.Function;
import com.gs.collections.api.block.predicate.Predicate;
import com.gs.collections.api.block.predicate.Predicate2;
import com.gs.collections.api.block.predicate.primitive.IntPredicate;
import com.gs.collections.api.block.procedure.Procedure;
import com.gs.collections.api.block.procedure.primitive.ObjectIntProcedure;
import com.gs.collections.api.map.ImmutableMap;
import com.gs.collections.api.map.MutableMap;
import com.gs.collections.api.multimap.MutableMultimap;
import com.gs.collections.api.multimap.bag.ImmutableBagMultimap;
import com.gs.collections.api.set.ImmutableSet;
import com.gs.collections.api.tuple.Pair;
import com.gs.collections.impl.bag.mutable.HashBag;
import com.gs.collections.impl.block.factory.Predicates;
import com.gs.collections.impl.block.factory.Predicates2;
import com.gs.collections.impl.block.procedure.FlatCollectProcedure;
import com.gs.collections.impl.block.procedure.MultimapEachPutProcedure;
import com.gs.collections.impl.block.procedure.MultimapPutProcedure;
import com.gs.collections.impl.factory.Bags;
import com.gs.collections.impl.map.mutable.UnifiedMap;
import com.gs.collections.impl.multimap.bag.HashBagMultimap;
import com.gs.collections.impl.set.mutable.UnifiedSet;
import com.gs.collections.impl.utility.ArrayIterate;
import com.gs.collections.impl.utility.Iterate;
import com.gs.collections.impl.utility.internal.IterableIterate;
/**
* @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.of(elements));
}
public static ImmutableArrayBag copyFrom(Bag bag)
{
int distinctItemCount = bag.sizeDistinct();
final T[] newKeys = (T[]) new Object[distinctItemCount];
final 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);
}
public int size()
{
int sum = 0;
for (int value : this.counts)
{
sum += value;
}
return sum;
}
public int sizeDistinct()
{
return this.keys.length;
}
public int occurrencesOf(Object item)
{
int index = ArrayIterate.detectIndexWith(this.keys, Predicates2.equal(), item);
if (index > -1)
{
return this.counts[index];
}
return 0;
}
public void forEachWithOccurrences(ObjectIntProcedure super T> objectIntProcedure)
{
for (int i = 0; i < this.keys.length; i++)
{
objectIntProcedure.value(this.keys[i], this.counts[i]);
}
}
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);
}
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;
}
public MutableMap toMapOfItemToCount()
{
final MutableMap map = UnifiedMap.newMap(this.size());
this.forEachWithOccurrences(new ObjectIntProcedure()
{
public void value(T item, int count)
{
map.put(item, count);
}
});
return map;
}
public ImmutableBag toImmutable()
{
return this;
}
public ImmutableBag newWithAll(Iterable extends T> elements)
{
return Bags.immutable.ofAll(Iterate.addAllTo(elements, HashBag.newBag(this)));
}
public ImmutableBag newWithoutAll(Iterable extends T> elements)
{
return this.reject(Predicates.in(elements));
}
public ImmutableBag selectByOccurrences(final IntPredicate predicate)
{
final MutableBag result = HashBag.newBag();
this.forEachWithOccurrences(new ObjectIntProcedure()
{
public void value(T each, int occurrences)
{
if (predicate.accept(occurrences))
{
result.addOccurrences(each, occurrences);
}
}
});
return result.toImmutable();
}
public ImmutableBag select(final Predicate super T> predicate)
{
final MutableBag result = HashBag.newBag();
this.forEachWithOccurrences(new ObjectIntProcedure()
{
public void value(T each, int occurrences)
{
if (predicate.accept(each))
{
result.addOccurrences(each, occurrences);
}
}
});
return result.toImmutable();
}
@Override
public > R selectWith(
Predicate2 super T, ? super P> predicate,
P parameter,
R targetCollection)
{
return IterableIterate.selectWith(this, predicate, parameter, targetCollection);
}
public ImmutableBag reject(Predicate super T> predicate)
{
return this.select(Predicates.not(predicate));
}
@Override
public > R rejectWith(
Predicate2 super T, ? super P> predicate,
P parameter,
R targetCollection)
{
return IterableIterate.rejectWith(this, predicate, parameter, targetCollection);
}
public ImmutableBag selectInstancesOf(final Class clazz)
{
final MutableBag result = HashBag.newBag();
this.forEachWithOccurrences(new ObjectIntProcedure()
{
public void value(T each, int index)
{
if (clazz.isInstance(each))
{
result.addOccurrences((S) each, index);
}
}
});
return ImmutableArrayBag.copyFrom(result);
}
public ImmutableBag collect(final Function super T, ? extends V> function)
{
final MutableBag result = HashBag.newBag();
this.forEachWithOccurrences(new ObjectIntProcedure()
{
public void value(T each, int index)
{
result.addOccurrences(function.valueOf(each), index);
}
});
return ImmutableArrayBag.copyFrom(result);
}
public ImmutableBag collectIf(
Predicate super T> predicate,
Function super T, ? extends V> function)
{
return ImmutableArrayBag.copyFrom(IterableIterate.collectIf(this, predicate, function, HashBag.newBag()));
}
public ImmutableBagMultimap groupBy(Function super T, ? extends V> function)
{
return this.groupBy(function, HashBagMultimap.newMultimap()).toImmutable();
}
public ImmutableBagMultimap groupByEach(Function super T, ? extends Iterable> function)
{
return this.groupByEach(function, HashBagMultimap.newMultimap()).toImmutable();
}
public ImmutableMap groupByUniqueKey(Function super T, ? extends V> function)
{
throw new UnsupportedOperationException(this.getClass().getSimpleName() + ".groupByUniqueKey() not implemented yet");
}
public T getFirst()
{
return ArrayIterate.getFirst(this.keys);
}
public T getLast()
{
return ArrayIterate.getLast(this.keys);
}
public ImmutableBag flatCollect(Function super T, ? extends Iterable> function)
{
FlatCollectProcedure procedure = new FlatCollectProcedure(function, HashBag.newBag());
this.forEach(procedure);
return ((MutableBag) procedure.getCollection()).toImmutable();
}
@Override
public > R groupBy(
Function super T, ? extends V> function, R target)
{
this.forEach(MultimapPutProcedure.on(target, function));
return target;
}
@Override
public > R groupByEach(
Function super T, ? extends Iterable> function, R target)
{
this.forEach(MultimapEachPutProcedure.on(target, function));
return target;
}
@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;
}
public void forEach(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);
}
}
}
public Iterator iterator()
{
return new ArrayBagIterator();
}
private final class ArrayBagIterator
implements Iterator
{
private int keyCount = -1;
private int countCount = -1;
private ArrayBagIterator()
{
this.keyCount = ImmutableArrayBag.this.keys.length - 1;
this.countCount = this.keyCount < 0 ? -1 : ImmutableArrayBag.this.counts[ImmutableArrayBag.this.keys.length - 1];
}
public boolean hasNext()
{
return this.keyCount >= 0;
}
public T next()
{
if (!this.hasNext())
{
throw new NoSuchElementException();
}
T result = ImmutableArrayBag.this.keys[this.keyCount];
--this.countCount;
if (this.countCount == 0)
{
--this.keyCount;
this.countCount = this.keyCount < 0 ? 0 : ImmutableArrayBag.this.counts[this.keyCount];
}
return result;
}
public void remove()
{
throw new UnsupportedOperationException("Cannot remove from an ImmutableArrayBag");
}
}
@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);
}
public ImmutableBag> zip(Iterable that)
{
return this.zip(that, HashBag.>newBag()).toImmutable();
}
public ImmutableSet> zipWithIndex()
{
return this.zipWithIndex(UnifiedSet.>newSet()).toImmutable();
}
protected Object writeReplace()
{
return new ImmutableBagSerializationProxy(this);
}
}