/*
* 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.multimap.bag.strategy;
import java.io.Externalizable;
import java.io.IOException;
import java.io.ObjectInput;
import java.io.ObjectOutput;
import com.gs.collections.api.bag.MutableBag;
import com.gs.collections.api.block.HashingStrategy;
import com.gs.collections.api.block.function.Function;
import com.gs.collections.api.block.predicate.Predicate2;
import com.gs.collections.api.map.MutableMap;
import com.gs.collections.api.multimap.Multimap;
import com.gs.collections.api.multimap.bag.MutableBagMultimap;
import com.gs.collections.api.tuple.Pair;
import com.gs.collections.impl.bag.mutable.HashBag;
import com.gs.collections.impl.map.strategy.mutable.UnifiedMapWithHashingStrategy;
import com.gs.collections.impl.multimap.bag.AbstractMutableBagMultimap;
import com.gs.collections.impl.utility.Iterate;
public final class HashBagMultimapWithHashingStrategy<K, V>
extends AbstractMutableBagMultimap<K, V> implements Externalizable
{
private static final long serialVersionUID = 1L;
private HashingStrategy<? super K> hashingStrategy;
/**
* @deprecated Empty default constructor used for serialization. Instantiating an HashBagMultimapWithHashingStrategy with
* this constructor will have a null multimapHashingStrategy, and throw NullPointerException when used.
*/
@SuppressWarnings("UnusedDeclaration")
@Deprecated
public HashBagMultimapWithHashingStrategy()
{
// For Externalizable use only
}
public HashBagMultimapWithHashingStrategy(HashingStrategy<? super K> hashingStrategy)
{
this.hashingStrategy = hashingStrategy;
this.map = this.createMap();
}
public HashBagMultimapWithHashingStrategy(HashBagMultimapWithHashingStrategy<K, V> multimap)
{
this(multimap.hashingStrategy, multimap);
}
public HashBagMultimapWithHashingStrategy(HashingStrategy<? super K> hashingStrategy, Multimap<? extends K, ? extends V> multimap)
{
this.hashingStrategy = hashingStrategy;
this.map = this.createMapWithKeyCount(Math.max(multimap.sizeDistinct() * 2, 16));
this.putAll(multimap);
}
public HashBagMultimapWithHashingStrategy(HashingStrategy<? super K> hashingStrategy, Pair<K, V>... pairs)
{
this(hashingStrategy);
this.putAllPairs(pairs);
}
public HashBagMultimapWithHashingStrategy(HashingStrategy<? super K> hashingStrategy, Iterable<Pair<K, V>> inputIterable)
{
this(hashingStrategy);
for (Pair<K, V> single : inputIterable)
{
this.add(single);
}
}
public static <K, V> HashBagMultimapWithHashingStrategy<K, V> newMultimap(HashBagMultimapWithHashingStrategy<K, V> multimap)
{
return new HashBagMultimapWithHashingStrategy<K, V>(multimap);
}
public static <K, V> HashBagMultimapWithHashingStrategy<K, V> newMultimap(HashingStrategy<? super K> multimapHashingStrategy, Multimap<? extends K, ? extends V> multimap)
{
return new HashBagMultimapWithHashingStrategy<K, V>(multimapHashingStrategy, multimap);
}
public static <K, V> HashBagMultimapWithHashingStrategy<K, V> newMultimap(HashingStrategy<? super K> multimapHashingStrategy)
{
return new HashBagMultimapWithHashingStrategy<K, V>(multimapHashingStrategy);
}
@SafeVarargs
public static <K, V> HashBagMultimapWithHashingStrategy<K, V> newMultimap(HashingStrategy<? super K> multimapHashingStrategy, Pair<K, V>... pairs)
{
return new HashBagMultimapWithHashingStrategy<K, V>(multimapHashingStrategy, pairs);
}
public static <K, V> HashBagMultimapWithHashingStrategy<K, V> newMultimap(HashingStrategy<? super K> multimapHashingStrategy, Iterable<Pair<K, V>> inputIterable)
{
return new HashBagMultimapWithHashingStrategy<K, V>(multimapHashingStrategy, inputIterable);
}
@Override
protected MutableMap<K, MutableBag<V>> createMap()
{
return UnifiedMapWithHashingStrategy.newMap(this.hashingStrategy);
}
@Override
protected MutableMap<K, MutableBag<V>> createMapWithKeyCount(int keyCount)
{
return UnifiedMapWithHashingStrategy.newMap(this.hashingStrategy, keyCount);
}
@Override
protected MutableBag<V> createCollection()
{
return HashBag.newBag();
}
public HashingStrategy<? super K> getKeyHashingStrategy()
{
return this.hashingStrategy;
}
public HashBagMultimapWithHashingStrategy<K, V> newEmpty()
{
return new HashBagMultimapWithHashingStrategy<K, V>(this.hashingStrategy);
}
@Override
public void writeExternal(ObjectOutput out) throws IOException
{
out.writeObject(this.hashingStrategy);
super.writeExternal(out);
}
@Override
public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException
{
this.hashingStrategy = (HashingStrategy<? super K>) in.readObject();
super.readExternal(in);
}
// Currently this returns a HashBagMultimap.
// On a future release, it will return HashBagWithHashingStrategyMultimap, where the HashBag collection hashing strategy
// will be the hashing strategy of this multimap
public MutableBagMultimap<V, K> flip()
{
return Iterate.flip(this);
}
public <V2> HashBagMultimapWithHashingStrategy<K, V2> collectValues(Function<? super V, ? extends V2> function)
{
return this.collectValues(function, HashBagMultimapWithHashingStrategy.<K, V2>newMultimap(this.hashingStrategy));
}
public HashBagMultimapWithHashingStrategy<K, V> selectKeysValues(Predicate2<? super K, ? super V> predicate)
{
return this.selectKeysValues(predicate, this.newEmpty());
}
public HashBagMultimapWithHashingStrategy<K, V> rejectKeysValues(Predicate2<? super K, ? super V> predicate)
{
return this.rejectKeysValues(predicate, this.newEmpty());
}
public HashBagMultimapWithHashingStrategy<K, V> selectKeysMultiValues(Predicate2<? super K, ? super Iterable<V>> predicate)
{
return this.selectKeysMultiValues(predicate, this.newEmpty());
}
public HashBagMultimapWithHashingStrategy<K, V> rejectKeysMultiValues(Predicate2<? super K, ? super Iterable<V>> predicate)
{
return this.rejectKeysMultiValues(predicate, this.newEmpty());
}
}