All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.eclipse.collections.impl.map.mutable.UnifiedMap Maven / Gradle / Ivy

There is a newer version: 12.0.0.M3
Show newest version
/*
 * 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.map.mutable;

import java.io.Externalizable;
import java.io.IOException;
import java.io.ObjectInput;
import java.io.ObjectInputStream;
import java.io.ObjectOutput;
import java.io.Serializable;
import java.lang.ref.WeakReference;
import java.lang.reflect.Array;
import java.util.Collection;
import java.util.Iterator;
import java.util.Map;
import java.util.NoSuchElementException;
import java.util.Optional;
import java.util.Set;

import org.eclipse.collections.api.block.function.Function;
import org.eclipse.collections.api.block.function.Function0;
import org.eclipse.collections.api.block.function.Function2;
import org.eclipse.collections.api.block.predicate.Predicate;
import org.eclipse.collections.api.block.predicate.Predicate2;
import org.eclipse.collections.api.block.procedure.Procedure;
import org.eclipse.collections.api.block.procedure.Procedure2;
import org.eclipse.collections.api.block.procedure.primitive.ObjectIntProcedure;
import org.eclipse.collections.api.map.ImmutableMap;
import org.eclipse.collections.api.map.MapIterable;
import org.eclipse.collections.api.map.MutableMap;
import org.eclipse.collections.api.map.UnsortedMapIterable;
import org.eclipse.collections.api.tuple.Pair;
import org.eclipse.collections.impl.block.factory.Functions;
import org.eclipse.collections.impl.block.factory.Predicates;
import org.eclipse.collections.impl.block.procedure.MapCollectProcedure;
import org.eclipse.collections.impl.factory.Maps;
import org.eclipse.collections.impl.factory.Sets;
import org.eclipse.collections.impl.list.mutable.FastList;
import org.eclipse.collections.impl.parallel.BatchIterable;
import org.eclipse.collections.impl.set.mutable.UnifiedSet;
import org.eclipse.collections.impl.tuple.ImmutableEntry;
import org.eclipse.collections.impl.tuple.Tuples;
import org.eclipse.collections.impl.utility.ArrayIterate;
import org.eclipse.collections.impl.utility.Iterate;

/**
 * UnifiedMap stores key/value pairs in a single array, where alternate slots are keys and values.  This is nicer to CPU caches as
 * consecutive memory addresses are very cheap to access.  Entry objects are not stored in the table like in java.util.HashMap.
 * Instead of trying to deal with collisions in the main array using Entry objects, we put a special object in
 * the key slot and put a regular Object[] in the value slot. The array contains the key value pairs in consecutive slots,
 * just like the main array, but it's a linear list with no hashing.
 * 

* The final result is a Map implementation that's leaner than java.util.HashMap and faster than Trove's THashMap. * The best of both approaches unified together, and thus the name UnifiedMap. */ @SuppressWarnings("ObjectEquality") public class UnifiedMap extends AbstractMutableMap implements Externalizable, BatchIterable { protected static final Object NULL_KEY = new Object() { @Override public boolean equals(Object obj) { throw new RuntimeException("Possible corruption through unsynchronized concurrent modification."); } @Override public int hashCode() { throw new RuntimeException("Possible corruption through unsynchronized concurrent modification."); } @Override public String toString() { return "UnifiedMap.NULL_KEY"; } }; protected static final Object CHAINED_KEY = new Object() { @Override public boolean equals(Object obj) { throw new RuntimeException("Possible corruption through unsynchronized concurrent modification."); } @Override public int hashCode() { throw new RuntimeException("Possible corruption through unsynchronized concurrent modification."); } @Override public String toString() { return "UnifiedMap.CHAINED_KEY"; } }; protected static final float DEFAULT_LOAD_FACTOR = 0.75f; protected static final int DEFAULT_INITIAL_CAPACITY = 8; private static final long serialVersionUID = 1L; protected transient Object[] table; protected transient int occupied; protected float loadFactor = DEFAULT_LOAD_FACTOR; protected int maxSize; public UnifiedMap() { this.allocate(DEFAULT_INITIAL_CAPACITY << 1); } public UnifiedMap(int initialCapacity) { this(initialCapacity, DEFAULT_LOAD_FACTOR); } public UnifiedMap(int initialCapacity, float loadFactor) { if (initialCapacity < 0) { throw new IllegalArgumentException("initial capacity cannot be less than 0"); } if (loadFactor <= 0.0) { throw new IllegalArgumentException("load factor cannot be less than or equal to 0"); } if (loadFactor > 1.0) { throw new IllegalArgumentException("load factor cannot be greater than 1"); } this.loadFactor = loadFactor; this.init(this.fastCeil(initialCapacity / loadFactor)); } public UnifiedMap(Map map) { this(Math.max(map.size(), DEFAULT_INITIAL_CAPACITY), DEFAULT_LOAD_FACTOR); this.putAll(map); } public UnifiedMap(Pair... pairs) { this(Math.max(pairs.length, DEFAULT_INITIAL_CAPACITY), DEFAULT_LOAD_FACTOR); ArrayIterate.forEach(pairs, new MapCollectProcedure, K, V>( this, Functions.firstOfPair(), Functions.secondOfPair())); } public static UnifiedMap newMap() { return new UnifiedMap<>(); } public static UnifiedMap newMap(int size) { return new UnifiedMap<>(size); } public static UnifiedMap newMap(int size, float loadFactor) { return new UnifiedMap<>(size, loadFactor); } public static UnifiedMap newMap(Map map) { return new UnifiedMap<>(map); } public static UnifiedMap newMapWith(Pair... pairs) { return new UnifiedMap<>(pairs); } public static UnifiedMap newMapWith(Iterable> inputIterable) { UnifiedMap outputMap = UnifiedMap.newMap(); for (Pair single : inputIterable) { outputMap.add(single); } return outputMap; } public static UnifiedMap newWithKeysValues(K key, V value) { return new UnifiedMap(1).withKeysValues(key, value); } public static UnifiedMap newWithKeysValues(K key1, V value1, K key2, V value2) { return new UnifiedMap(2).withKeysValues(key1, value1, key2, value2); } public static UnifiedMap newWithKeysValues(K key1, V value1, K key2, V value2, K key3, V value3) { return new UnifiedMap(3).withKeysValues(key1, value1, key2, value2, key3, value3); } public static UnifiedMap newWithKeysValues( K key1, V value1, K key2, V value2, K key3, V value3, K key4, V value4) { return new UnifiedMap(4).withKeysValues(key1, value1, key2, value2, key3, value3, key4, value4); } public UnifiedMap withKeysValues(K key, V value) { this.put(key, value); return this; } public UnifiedMap withKeysValues(K key1, V value1, K key2, V value2) { this.put(key1, value1); this.put(key2, value2); return this; } public UnifiedMap withKeysValues(K key1, V value1, K key2, V value2, K key3, V value3) { this.put(key1, value1); this.put(key2, value2); this.put(key3, value3); return this; } public UnifiedMap withKeysValues(K key1, V value1, K key2, V value2, K key3, V value3, K key4, V value4) { this.put(key1, value1); this.put(key2, value2); this.put(key3, value3); this.put(key4, value4); return this; } @Override public UnifiedMap clone() { return new UnifiedMap<>(this); } @Override public MutableMap newEmpty() { return new UnifiedMap<>(); } @Override public MutableMap newEmpty(int capacity) { return UnifiedMap.newMap(capacity); } private int fastCeil(float v) { int possibleResult = (int) v; if (v - possibleResult > 0.0F) { possibleResult++; } return possibleResult; } protected int init(int initialCapacity) { int capacity = 1; while (capacity < initialCapacity) { capacity <<= 1; } return this.allocate(capacity); } protected int allocate(int capacity) { this.allocateTable(capacity << 1); // the table size is twice the capacity to handle both keys and values this.computeMaxSize(capacity); return capacity; } protected void allocateTable(int sizeToAllocate) { this.table = new Object[sizeToAllocate]; } protected void computeMaxSize(int capacity) { this.maxSize = Math.min(capacity - 1, (int) (capacity * this.loadFactor)); } protected final int index(Object key) { // This function ensures that hashCodes that differ only by // constant multiples at each bit position have a bounded // number of collisions (approximately 8 at default load factor). int h = key == null ? 0 : key.hashCode(); h ^= h >>> 20 ^ h >>> 12; h ^= h >>> 7 ^ h >>> 4; return (h & (this.table.length >> 1) - 1) << 1; } @Override public void clear() { if (this.occupied == 0) { return; } this.occupied = 0; Object[] set = this.table; for (int i = set.length; i-- > 0; ) { set[i] = null; } } @Override public V put(K key, V value) { int index = this.index(key); Object cur = this.table[index]; if (cur == null) { this.table[index] = UnifiedMap.toSentinelIfNull(key); this.table[index + 1] = value; if (++this.occupied > this.maxSize) { this.rehash(this.table.length); } return null; } if (cur != CHAINED_KEY && this.nonNullTableObjectEquals(cur, key)) { V result = (V) this.table[index + 1]; this.table[index + 1] = value; return result; } return this.chainedPut(key, index, value); } private V chainedPut(K key, int index, V value) { if (this.table[index] == CHAINED_KEY) { Object[] chain = (Object[]) this.table[index + 1]; for (int i = 0; i < chain.length; i += 2) { if (chain[i] == null) { chain[i] = UnifiedMap.toSentinelIfNull(key); chain[i + 1] = value; if (++this.occupied > this.maxSize) { this.rehash(this.table.length); } return null; } if (this.nonNullTableObjectEquals(chain[i], key)) { V result = (V) chain[i + 1]; chain[i + 1] = value; return result; } } Object[] newChain = new Object[chain.length + 4]; System.arraycopy(chain, 0, newChain, 0, chain.length); this.table[index + 1] = newChain; newChain[chain.length] = UnifiedMap.toSentinelIfNull(key); newChain[chain.length + 1] = value; if (++this.occupied > this.maxSize) { this.rehash(this.table.length); } return null; } Object[] newChain = new Object[4]; newChain[0] = this.table[index]; newChain[1] = this.table[index + 1]; newChain[2] = UnifiedMap.toSentinelIfNull(key); newChain[3] = value; this.table[index] = CHAINED_KEY; this.table[index + 1] = newChain; if (++this.occupied > this.maxSize) { this.rehash(this.table.length); } return null; } @Override public V updateValue(K key, Function0 factory, Function function) { int index = this.index(key); Object cur = this.table[index]; if (cur == null) { this.table[index] = UnifiedMap.toSentinelIfNull(key); V result = function.valueOf(factory.value()); this.table[index + 1] = result; ++this.occupied; return result; } if (cur != CHAINED_KEY && this.nonNullTableObjectEquals(cur, key)) { V oldValue = (V) this.table[index + 1]; V newValue = function.valueOf(oldValue); this.table[index + 1] = newValue; return newValue; } return this.chainedUpdateValue(key, index, factory, function); } private V chainedUpdateValue(K key, int index, Function0 factory, Function function) { if (this.table[index] == CHAINED_KEY) { Object[] chain = (Object[]) this.table[index + 1]; for (int i = 0; i < chain.length; i += 2) { if (chain[i] == null) { chain[i] = UnifiedMap.toSentinelIfNull(key); V result = function.valueOf(factory.value()); chain[i + 1] = result; if (++this.occupied > this.maxSize) { this.rehash(this.table.length); } return result; } if (this.nonNullTableObjectEquals(chain[i], key)) { V oldValue = (V) chain[i + 1]; V result = function.valueOf(oldValue); chain[i + 1] = result; return result; } } Object[] newChain = new Object[chain.length + 4]; System.arraycopy(chain, 0, newChain, 0, chain.length); this.table[index + 1] = newChain; newChain[chain.length] = UnifiedMap.toSentinelIfNull(key); V result = function.valueOf(factory.value()); newChain[chain.length + 1] = result; if (++this.occupied > this.maxSize) { this.rehash(this.table.length); } return result; } Object[] newChain = new Object[4]; newChain[0] = this.table[index]; newChain[1] = this.table[index + 1]; newChain[2] = UnifiedMap.toSentinelIfNull(key); V result = function.valueOf(factory.value()); newChain[3] = result; this.table[index] = CHAINED_KEY; this.table[index + 1] = newChain; if (++this.occupied > this.maxSize) { this.rehash(this.table.length); } return result; } @Override public

V updateValueWith(K key, Function0 factory, Function2 function, P parameter) { int index = this.index(key); Object cur = this.table[index]; if (cur == null) { this.table[index] = UnifiedMap.toSentinelIfNull(key); V result = function.value(factory.value(), parameter); this.table[index + 1] = result; ++this.occupied; return result; } if (cur != CHAINED_KEY && this.nonNullTableObjectEquals(cur, key)) { V oldValue = (V) this.table[index + 1]; V newValue = function.value(oldValue, parameter); this.table[index + 1] = newValue; return newValue; } return this.chainedUpdateValueWith(key, index, factory, function, parameter); } private

V chainedUpdateValueWith( K key, int index, Function0 factory, Function2 function, P parameter) { if (this.table[index] == CHAINED_KEY) { Object[] chain = (Object[]) this.table[index + 1]; for (int i = 0; i < chain.length; i += 2) { if (chain[i] == null) { chain[i] = UnifiedMap.toSentinelIfNull(key); V result = function.value(factory.value(), parameter); chain[i + 1] = result; if (++this.occupied > this.maxSize) { this.rehash(this.table.length); } return result; } if (this.nonNullTableObjectEquals(chain[i], key)) { V oldValue = (V) chain[i + 1]; V result = function.value(oldValue, parameter); chain[i + 1] = result; return result; } } Object[] newChain = new Object[chain.length + 4]; System.arraycopy(chain, 0, newChain, 0, chain.length); this.table[index + 1] = newChain; newChain[chain.length] = UnifiedMap.toSentinelIfNull(key); V result = function.value(factory.value(), parameter); newChain[chain.length + 1] = result; if (++this.occupied > this.maxSize) { this.rehash(this.table.length); } return result; } Object[] newChain = new Object[4]; newChain[0] = this.table[index]; newChain[1] = this.table[index + 1]; newChain[2] = UnifiedMap.toSentinelIfNull(key); V result = function.value(factory.value(), parameter); newChain[3] = result; this.table[index] = CHAINED_KEY; this.table[index + 1] = newChain; if (++this.occupied > this.maxSize) { this.rehash(this.table.length); } return result; } @Override public V getIfAbsentPut(K key, Function0 function) { int index = this.index(key); Object cur = this.table[index]; if (cur == null) { V result = function.value(); this.table[index] = UnifiedMap.toSentinelIfNull(key); this.table[index + 1] = result; if (++this.occupied > this.maxSize) { this.rehash(this.table.length); } return result; } if (cur != CHAINED_KEY && this.nonNullTableObjectEquals(cur, key)) { return (V) this.table[index + 1]; } return this.chainedGetIfAbsentPut(key, index, function); } private V chainedGetIfAbsentPut(K key, int index, Function0 function) { V result = null; if (this.table[index] == CHAINED_KEY) { Object[] chain = (Object[]) this.table[index + 1]; int i = 0; for (; i < chain.length; i += 2) { if (chain[i] == null) { result = function.value(); chain[i] = UnifiedMap.toSentinelIfNull(key); chain[i + 1] = result; if (++this.occupied > this.maxSize) { this.rehash(this.table.length); } break; } if (this.nonNullTableObjectEquals(chain[i], key)) { result = (V) chain[i + 1]; break; } } if (i == chain.length) { result = function.value(); Object[] newChain = new Object[chain.length + 4]; System.arraycopy(chain, 0, newChain, 0, chain.length); newChain[i] = UnifiedMap.toSentinelIfNull(key); newChain[i + 1] = result; this.table[index + 1] = newChain; if (++this.occupied > this.maxSize) { this.rehash(this.table.length); } } } else { result = function.value(); Object[] newChain = new Object[4]; newChain[0] = this.table[index]; newChain[1] = this.table[index + 1]; newChain[2] = UnifiedMap.toSentinelIfNull(key); newChain[3] = result; this.table[index] = CHAINED_KEY; this.table[index + 1] = newChain; if (++this.occupied > this.maxSize) { this.rehash(this.table.length); } } return result; } @Override public V getIfAbsentPut(K key, V value) { int index = this.index(key); Object cur = this.table[index]; if (cur == null) { this.table[index] = UnifiedMap.toSentinelIfNull(key); this.table[index + 1] = value; if (++this.occupied > this.maxSize) { this.rehash(this.table.length); } return value; } if (cur != CHAINED_KEY && this.nonNullTableObjectEquals(cur, key)) { return (V) this.table[index + 1]; } return this.chainedGetIfAbsentPut(key, index, value); } private V chainedGetIfAbsentPut(K key, int index, V value) { V result = value; if (this.table[index] == CHAINED_KEY) { Object[] chain = (Object[]) this.table[index + 1]; int i = 0; for (; i < chain.length; i += 2) { if (chain[i] == null) { chain[i] = UnifiedMap.toSentinelIfNull(key); chain[i + 1] = value; if (++this.occupied > this.maxSize) { this.rehash(this.table.length); } break; } if (this.nonNullTableObjectEquals(chain[i], key)) { result = (V) chain[i + 1]; break; } } if (i == chain.length) { Object[] newChain = new Object[chain.length + 4]; System.arraycopy(chain, 0, newChain, 0, chain.length); newChain[i] = UnifiedMap.toSentinelIfNull(key); newChain[i + 1] = value; this.table[index + 1] = newChain; if (++this.occupied > this.maxSize) { this.rehash(this.table.length); } } } else { Object[] newChain = new Object[4]; newChain[0] = this.table[index]; newChain[1] = this.table[index + 1]; newChain[2] = UnifiedMap.toSentinelIfNull(key); newChain[3] = value; this.table[index] = CHAINED_KEY; this.table[index + 1] = newChain; if (++this.occupied > this.maxSize) { this.rehash(this.table.length); } } return result; } @Override public

V getIfAbsentPutWith(K key, Function function, P parameter) { int index = this.index(key); Object cur = this.table[index]; if (cur == null) { V result = function.valueOf(parameter); this.table[index] = UnifiedMap.toSentinelIfNull(key); this.table[index + 1] = result; if (++this.occupied > this.maxSize) { this.rehash(this.table.length); } return result; } if (cur != CHAINED_KEY && this.nonNullTableObjectEquals(cur, key)) { return (V) this.table[index + 1]; } return this.chainedGetIfAbsentPutWith(key, index, function, parameter); } private

V chainedGetIfAbsentPutWith(K key, int index, Function function, P parameter) { V result = null; if (this.table[index] == CHAINED_KEY) { Object[] chain = (Object[]) this.table[index + 1]; int i = 0; for (; i < chain.length; i += 2) { if (chain[i] == null) { result = function.valueOf(parameter); chain[i] = UnifiedMap.toSentinelIfNull(key); chain[i + 1] = result; if (++this.occupied > this.maxSize) { this.rehash(this.table.length); } break; } if (this.nonNullTableObjectEquals(chain[i], key)) { result = (V) chain[i + 1]; break; } } if (i == chain.length) { result = function.valueOf(parameter); Object[] newChain = new Object[chain.length + 4]; System.arraycopy(chain, 0, newChain, 0, chain.length); newChain[i] = UnifiedMap.toSentinelIfNull(key); newChain[i + 1] = result; this.table[index + 1] = newChain; if (++this.occupied > this.maxSize) { this.rehash(this.table.length); } } } else { result = function.valueOf(parameter); Object[] newChain = new Object[4]; newChain[0] = this.table[index]; newChain[1] = this.table[index + 1]; newChain[2] = UnifiedMap.toSentinelIfNull(key); newChain[3] = result; this.table[index] = CHAINED_KEY; this.table[index + 1] = newChain; if (++this.occupied > this.maxSize) { this.rehash(this.table.length); } } return result; } public int getCollidingBuckets() { int count = 0; for (int i = 0; i < this.table.length; i += 2) { if (this.table[i] == CHAINED_KEY) { count++; } } return count; } /** * Returns the number of JVM words that is used by this map. A word is 4 bytes in a 32bit VM and 8 bytes in a 64bit * VM. Each array has a 2 word header, thus the formula is: * words = (internal table length + 2) + sum (for all chains (chain length + 2)) * * @return the number of JVM words that is used by this map. */ public int getMapMemoryUsedInWords() { int headerSize = 2; int sizeInWords = this.table.length + headerSize; for (int i = 0; i < this.table.length; i += 2) { if (this.table[i] == CHAINED_KEY) { sizeInWords += headerSize + ((Object[]) this.table[i + 1]).length; } } return sizeInWords; } protected void rehash(int newCapacity) { int oldLength = this.table.length; Object[] old = this.table; this.allocate(newCapacity); this.occupied = 0; for (int i = 0; i < oldLength; i += 2) { Object cur = old[i]; if (cur == CHAINED_KEY) { Object[] chain = (Object[]) old[i + 1]; for (int j = 0; j < chain.length; j += 2) { if (chain[j] != null) { this.put(this.nonSentinel(chain[j]), (V) chain[j + 1]); } } } else if (cur != null) { this.put(this.nonSentinel(cur), (V) old[i + 1]); } } } @Override public V get(Object key) { int index = this.index(key); Object cur = this.table[index]; if (cur != null) { Object val = this.table[index + 1]; if (cur == CHAINED_KEY) { return this.getFromChain((Object[]) val, (K) key); } if (this.nonNullTableObjectEquals(cur, (K) key)) { return (V) val; } } return null; } private V getFromChain(Object[] chain, K key) { for (int i = 0; i < chain.length; i += 2) { Object k = chain[i]; if (k == null) { return null; } if (this.nonNullTableObjectEquals(k, key)) { return (V) chain[i + 1]; } } return null; } @Override public boolean containsKey(Object key) { int index = this.index(key); Object cur = this.table[index]; if (cur == null) { return false; } if (cur != CHAINED_KEY && this.nonNullTableObjectEquals(cur, (K) key)) { return true; } return cur == CHAINED_KEY && this.chainContainsKey((Object[]) this.table[index + 1], (K) key); } private boolean chainContainsKey(Object[] chain, K key) { for (int i = 0; i < chain.length; i += 2) { Object k = chain[i]; if (k == null) { return false; } if (this.nonNullTableObjectEquals(k, key)) { return true; } } return false; } @Override public boolean containsValue(Object value) { for (int i = 0; i < this.table.length; i += 2) { if (this.table[i] == CHAINED_KEY) { if (this.chainedContainsValue((Object[]) this.table[i + 1], (V) value)) { return true; } } else if (this.table[i] != null) { if (UnifiedMap.nullSafeEquals(value, this.table[i + 1])) { return true; } } } return false; } private boolean chainedContainsValue(Object[] chain, V value) { for (int i = 0; i < chain.length; i += 2) { if (chain[i] == null) { return false; } if (UnifiedMap.nullSafeEquals(value, chain[i + 1])) { return true; } } return false; } @Override public void forEachKeyValue(Procedure2 procedure) { for (int i = 0; i < this.table.length; i += 2) { Object cur = this.table[i]; if (cur == CHAINED_KEY) { this.chainedForEachEntry((Object[]) this.table[i + 1], procedure); } else if (cur != null) { procedure.value(this.nonSentinel(cur), (V) this.table[i + 1]); } } } @Override public V getFirst() { for (int i = 0; i < this.table.length; i += 2) { Object cur = this.table[i]; if (cur == CHAINED_KEY) { Object[] chain = (Object[]) this.table[i + 1]; return (V) chain[1]; } if (cur != null) { return (V) this.table[i + 1]; } } return null; } @Override public MutableMap collectKeysAndValues( Iterable iterable, Function keyFunction, Function valueFunction) { Iterate.forEach(iterable, new MapCollectProcedure<>(this, keyFunction, valueFunction)); return this; } @Override public V removeKey(K key) { return this.remove(key); } private void chainedForEachEntry(Object[] chain, Procedure2 procedure) { for (int i = 0; i < chain.length; i += 2) { Object cur = chain[i]; if (cur == null) { return; } procedure.value(this.nonSentinel(cur), (V) chain[i + 1]); } } @Override public int getBatchCount(int batchSize) { return Math.max(1, this.table.length / 2 / batchSize); } @Override public void batchForEach(Procedure procedure, int sectionIndex, int sectionCount) { int sectionSize = this.table.length / sectionCount; int start = sectionIndex * sectionSize; int end = sectionIndex == sectionCount - 1 ? this.table.length : start + sectionSize; if (start % 2 == 0) { start++; } for (int i = start; i < end; i += 2) { Object value = this.table[i]; if (value instanceof Object[]) { this.chainedForEachValue((Object[]) value, procedure); } else if (value == null && this.table[i - 1] != null || value != null) { procedure.value((V) value); } } } @Override public void forEachKey(Procedure procedure) { for (int i = 0; i < this.table.length; i += 2) { Object cur = this.table[i]; if (cur == CHAINED_KEY) { this.chainedForEachKey((Object[]) this.table[i + 1], procedure); } else if (cur != null) { procedure.value(this.nonSentinel(cur)); } } } private void chainedForEachKey(Object[] chain, Procedure procedure) { for (int i = 0; i < chain.length; i += 2) { Object cur = chain[i]; if (cur == null) { return; } procedure.value(this.nonSentinel(cur)); } } @Override public void forEachValue(Procedure procedure) { for (int i = 0; i < this.table.length; i += 2) { Object cur = this.table[i]; if (cur == CHAINED_KEY) { this.chainedForEachValue((Object[]) this.table[i + 1], procedure); } else if (cur != null) { procedure.value((V) this.table[i + 1]); } } } private void chainedForEachValue(Object[] chain, Procedure procedure) { for (int i = 0; i < chain.length; i += 2) { Object cur = chain[i]; if (cur == null) { return; } procedure.value((V) chain[i + 1]); } } @Override public boolean isEmpty() { return this.occupied == 0; } @Override public void putAll(Map map) { if (map instanceof UnifiedMap) { this.copyMap((UnifiedMap) map); } else if (map instanceof UnsortedMapIterable) { MapIterable mapIterable = (MapIterable) map; mapIterable.forEachKeyValue(this::put); } else { Iterator> iterator = this.getEntrySetFrom(map).iterator(); while (iterator.hasNext()) { Entry entry = iterator.next(); this.put(entry.getKey(), entry.getValue()); } } } private Set> getEntrySetFrom(Map map) { Set> entries = map.entrySet(); if (entries != null) { return entries; } if (map.isEmpty()) { return Sets.immutable.>of().castToSet(); } throw new IllegalStateException("Entry set was null and size was non-zero"); } protected void copyMap(UnifiedMap unifiedMap) { for (int i = 0; i < unifiedMap.table.length; i += 2) { Object cur = unifiedMap.table[i]; if (cur == CHAINED_KEY) { this.copyChain((Object[]) unifiedMap.table[i + 1]); } else if (cur != null) { this.put(this.nonSentinel(cur), (V) unifiedMap.table[i + 1]); } } } private void copyChain(Object[] chain) { for (int j = 0; j < chain.length; j += 2) { Object cur = chain[j]; if (cur == null) { break; } this.put(this.nonSentinel(cur), (V) chain[j + 1]); } } @Override public V remove(Object key) { int index = this.index(key); Object cur = this.table[index]; if (cur != null) { Object val = this.table[index + 1]; if (cur == CHAINED_KEY) { return this.removeFromChain((Object[]) val, (K) key, index); } if (this.nonNullTableObjectEquals(cur, (K) key)) { this.table[index] = null; this.table[index + 1] = null; this.occupied--; return (V) val; } } return null; } private V removeFromChain(Object[] chain, K key, int index) { for (int i = 0; i < chain.length; i += 2) { Object k = chain[i]; if (k == null) { return null; } if (this.nonNullTableObjectEquals(k, key)) { V val = (V) chain[i + 1]; this.overwriteWithLastElementFromChain(chain, index, i); return val; } } return null; } private void overwriteWithLastElementFromChain(Object[] chain, int index, int i) { int j = chain.length - 2; for (; j > i; j -= 2) { if (chain[j] != null) { chain[i] = chain[j]; chain[i + 1] = chain[j + 1]; break; } } chain[j] = null; chain[j + 1] = null; if (j == 0) { this.table[index] = null; this.table[index + 1] = null; } this.occupied--; } @Override public int size() { return this.occupied; } @Override public Set> entrySet() { return new EntrySet(); } @Override public Set keySet() { return new KeySet(); } @Override public Collection values() { return new ValuesCollection(); } @Override public boolean equals(Object object) { if (this == object) { return true; } if (!(object instanceof Map)) { return false; } Map other = (Map) object; if (this.size() != other.size()) { return false; } for (int i = 0; i < this.table.length; i += 2) { Object cur = this.table[i]; if (cur == CHAINED_KEY) { if (!this.chainedEquals((Object[]) this.table[i + 1], other)) { return false; } } else if (cur != null) { K key = this.nonSentinel(cur); V value = (V) this.table[i + 1]; Object otherValue = other.get(key); if (!UnifiedMap.nullSafeEquals(otherValue, value) || (value == null && otherValue == null && !other.containsKey(key))) { return false; } } } return true; } private boolean chainedEquals(Object[] chain, Map other) { for (int i = 0; i < chain.length; i += 2) { Object cur = chain[i]; if (cur == null) { return true; } K key = this.nonSentinel(cur); V value = (V) chain[i + 1]; Object otherValue = other.get(key); if (!UnifiedMap.nullSafeEquals(otherValue, value) || (value == null && otherValue == null && !other.containsKey(key))) { return false; } } return true; } @Override public int hashCode() { int hashCode = 0; for (int i = 0; i < this.table.length; i += 2) { Object cur = this.table[i]; if (cur == CHAINED_KEY) { hashCode += this.chainedHashCode((Object[]) this.table[i + 1]); } else if (cur != null) { Object value = this.table[i + 1]; hashCode += (cur == NULL_KEY ? 0 : cur.hashCode()) ^ (value == null ? 0 : value.hashCode()); } } return hashCode; } private int chainedHashCode(Object[] chain) { int hashCode = 0; for (int i = 0; i < chain.length; i += 2) { Object cur = chain[i]; if (cur == null) { return hashCode; } Object value = chain[i + 1]; hashCode += (cur == NULL_KEY ? 0 : cur.hashCode()) ^ (value == null ? 0 : value.hashCode()); } return hashCode; } @Override public String toString() { StringBuilder builder = new StringBuilder(); builder.append('{'); this.forEachKeyValue(new Procedure2() { private boolean first = true; public void value(K key, V value) { if (this.first) { this.first = false; } else { builder.append(", "); } builder.append(key == UnifiedMap.this ? "(this Map)" : key); builder.append('='); builder.append(value == UnifiedMap.this ? "(this Map)" : value); } }); builder.append('}'); return builder.toString(); } public boolean trimToSize() { if (this.table.length <= this.fastCeil(this.occupied / this.loadFactor) << 2) { return false; } Object[] temp = this.table; this.init(this.fastCeil(this.occupied / this.loadFactor)); if (this.isEmpty()) { return true; } int mask = this.table.length - 1; for (int j = 0; j < temp.length; j += 2) { Object key = temp[j]; if (key == CHAINED_KEY) { Object[] chain = (Object[]) temp[j + 1]; for (int i = 0; i < chain.length; i += 2) { Object cur = chain[i]; if (cur != null) { this.putForTrim((K) cur, (V) chain[i + 1], j, mask); } } } else if (key != null) { this.putForTrim((K) key, (V) temp[j + 1], j, mask); } } return true; } private void putForTrim(K key, V value, int oldIndex, int mask) { int index = oldIndex & mask; Object cur = this.table[index]; if (cur == null) { this.table[index] = key; this.table[index + 1] = value; return; } this.chainedPutForTrim(key, index, value); } private void chainedPutForTrim(K key, int index, V value) { if (this.table[index] == CHAINED_KEY) { Object[] chain = (Object[]) this.table[index + 1]; for (int i = 0; i < chain.length; i += 2) { if (chain[i] == null) { chain[i] = key; chain[i + 1] = value; return; } } Object[] newChain = new Object[chain.length + 4]; System.arraycopy(chain, 0, newChain, 0, chain.length); this.table[index + 1] = newChain; newChain[chain.length] = UnifiedMap.toSentinelIfNull(key); newChain[chain.length + 1] = value; return; } Object[] newChain = new Object[4]; newChain[0] = this.table[index]; newChain[1] = this.table[index + 1]; newChain[2] = key; newChain[3] = value; this.table[index] = CHAINED_KEY; this.table[index + 1] = newChain; } @Override public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException { int size = in.readInt(); this.loadFactor = in.readFloat(); this.init(Math.max((int) (size / this.loadFactor) + 1, DEFAULT_INITIAL_CAPACITY)); for (int i = 0; i < size; i++) { this.put((K) in.readObject(), (V) in.readObject()); } } @Override public void writeExternal(ObjectOutput out) throws IOException { out.writeInt(this.size()); out.writeFloat(this.loadFactor); for (int i = 0; i < this.table.length; i += 2) { Object o = this.table[i]; if (o != null) { if (o == CHAINED_KEY) { this.writeExternalChain(out, (Object[]) this.table[i + 1]); } else { out.writeObject(this.nonSentinel(o)); out.writeObject(this.table[i + 1]); } } } } private void writeExternalChain(ObjectOutput out, Object[] chain) throws IOException { for (int i = 0; i < chain.length; i += 2) { Object cur = chain[i]; if (cur == null) { return; } out.writeObject(this.nonSentinel(cur)); out.writeObject(chain[i + 1]); } } @Override public void forEachWithIndex(ObjectIntProcedure objectIntProcedure) { int index = 0; for (int i = 0; i < this.table.length; i += 2) { Object cur = this.table[i]; if (cur == CHAINED_KEY) { index = this.chainedForEachValueWithIndex((Object[]) this.table[i + 1], objectIntProcedure, index); } else if (cur != null) { objectIntProcedure.value((V) this.table[i + 1], index++); } } } private int chainedForEachValueWithIndex(Object[] chain, ObjectIntProcedure objectIntProcedure, int index) { for (int i = 0; i < chain.length; i += 2) { Object cur = chain[i]; if (cur == null) { return index; } objectIntProcedure.value((V) chain[i + 1], index++); } return index; } @Override public

void forEachWith(Procedure2 procedure, P parameter) { for (int i = 0; i < this.table.length; i += 2) { Object cur = this.table[i]; if (cur == CHAINED_KEY) { this.chainedForEachValueWith((Object[]) this.table[i + 1], procedure, parameter); } else if (cur != null) { procedure.value((V) this.table[i + 1], parameter); } } } private

void chainedForEachValueWith( Object[] chain, Procedure2 procedure, P parameter) { for (int i = 0; i < chain.length; i += 2) { Object cur = chain[i]; if (cur == null) { return; } procedure.value((V) chain[i + 1], parameter); } } @Override public MutableMap collectValues(Function2 function) { UnifiedMap target = UnifiedMap.newMap(); target.loadFactor = this.loadFactor; target.occupied = this.occupied; target.allocate(this.table.length >> 1); for (int i = 0; i < target.table.length; i += 2) { target.table[i] = this.table[i]; if (this.table[i] == CHAINED_KEY) { Object[] chainedTable = (Object[]) this.table[i + 1]; Object[] chainedTargetTable = new Object[chainedTable.length]; for (int j = 0; j < chainedTargetTable.length; j += 2) { if (chainedTable[j] != null) { chainedTargetTable[j] = chainedTable[j]; chainedTargetTable[j + 1] = function.value(this.nonSentinel(chainedTable[j]), (V) chainedTable[j + 1]); } } target.table[i + 1] = chainedTargetTable; } else if (this.table[i] != null) { target.table[i + 1] = function.value(this.nonSentinel(this.table[i]), (V) this.table[i + 1]); } } return target; } @Override public Pair detect(Predicate2 predicate) { for (int i = 0; i < this.table.length; i += 2) { if (this.table[i] == CHAINED_KEY) { Object[] chainedTable = (Object[]) this.table[i + 1]; for (int j = 0; j < chainedTable.length; j += 2) { if (chainedTable[j] != null) { K key = this.nonSentinel(chainedTable[j]); V value = (V) chainedTable[j + 1]; if (predicate.accept(key, value)) { return Tuples.pair(key, value); } } } } else if (this.table[i] != null) { K key = this.nonSentinel(this.table[i]); V value = (V) this.table[i + 1]; if (predicate.accept(key, value)) { return Tuples.pair(key, value); } } } return null; } @Override public V detect(Predicate predicate) { for (int i = 0; i < this.table.length; i += 2) { if (this.table[i] == CHAINED_KEY) { Object[] chainedTable = (Object[]) this.table[i + 1]; for (int j = 0; j < chainedTable.length; j += 2) { if (chainedTable[j] != null) { V value = (V) chainedTable[j + 1]; if (predicate.accept(value)) { return value; } } } } else if (this.table[i] != null) { V value = (V) this.table[i + 1]; if (predicate.accept(value)) { return value; } } } return null; } @Override public

V detectWith(Predicate2 predicate, P parameter) { for (int i = 0; i < this.table.length; i += 2) { if (this.table[i] == CHAINED_KEY) { Object[] chainedTable = (Object[]) this.table[i + 1]; for (int j = 0; j < chainedTable.length; j += 2) { if (chainedTable[j] != null) { V value = (V) chainedTable[j + 1]; if (predicate.accept(value, parameter)) { return value; } } } } else if (this.table[i] != null) { V value = (V) this.table[i + 1]; if (predicate.accept(value, parameter)) { return value; } } } return null; } @Override public Optional detectOptional(Predicate predicate) { for (int i = 0; i < this.table.length; i += 2) { if (this.table[i] == CHAINED_KEY) { Object[] chainedTable = (Object[]) this.table[i + 1]; for (int j = 0; j < chainedTable.length; j += 2) { if (chainedTable[j] != null) { V value = (V) chainedTable[j + 1]; if (predicate.accept(value)) { return Optional.of(value); } } } } else if (this.table[i] != null) { V value = (V) this.table[i + 1]; if (predicate.accept(value)) { return Optional.of(value); } } } return Optional.empty(); } @Override public

Optional detectWithOptional(Predicate2 predicate, P parameter) { for (int i = 0; i < this.table.length; i += 2) { if (this.table[i] == CHAINED_KEY) { Object[] chainedTable = (Object[]) this.table[i + 1]; for (int j = 0; j < chainedTable.length; j += 2) { if (chainedTable[j] != null) { V value = (V) chainedTable[j + 1]; if (predicate.accept(value, parameter)) { return Optional.of(value); } } } } else if (this.table[i] != null) { V value = (V) this.table[i + 1]; if (predicate.accept(value, parameter)) { return Optional.of(value); } } } return Optional.empty(); } @Override public V detectIfNone(Predicate predicate, Function0 function) { for (int i = 0; i < this.table.length; i += 2) { if (this.table[i] == CHAINED_KEY) { Object[] chainedTable = (Object[]) this.table[i + 1]; for (int j = 0; j < chainedTable.length; j += 2) { if (chainedTable[j] != null) { V value = (V) chainedTable[j + 1]; if (predicate.accept(value)) { return value; } } } } else if (this.table[i] != null) { V value = (V) this.table[i + 1]; if (predicate.accept(value)) { return value; } } } return function.value(); } @Override public

V detectWithIfNone( Predicate2 predicate, P parameter, Function0 function) { for (int i = 0; i < this.table.length; i += 2) { if (this.table[i] == CHAINED_KEY) { Object[] chainedTable = (Object[]) this.table[i + 1]; for (int j = 0; j < chainedTable.length; j += 2) { if (chainedTable[j] != null) { V value = (V) chainedTable[j + 1]; if (predicate.accept(value, parameter)) { return value; } } } } else if (this.table[i] != null) { V value = (V) this.table[i + 1]; if (predicate.accept(value, parameter)) { return value; } } } return function.value(); } private boolean shortCircuit( Predicate predicate, boolean expected, boolean onShortCircuit, boolean atEnd) { for (int i = 0; i < this.table.length; i += 2) { if (this.table[i] == CHAINED_KEY) { Object[] chainedTable = (Object[]) this.table[i + 1]; for (int j = 0; j < chainedTable.length; j += 2) { if (chainedTable[j] != null) { V value = (V) chainedTable[j + 1]; if (predicate.accept(value) == expected) { return onShortCircuit; } } } } else if (this.table[i] != null) { V value = (V) this.table[i + 1]; if (predicate.accept(value) == expected) { return onShortCircuit; } } } return atEnd; } private

boolean shortCircuitWith( Predicate2 predicate, P parameter, boolean expected, boolean onShortCircuit, boolean atEnd) { for (int i = 0; i < this.table.length; i += 2) { if (this.table[i] == CHAINED_KEY) { Object[] chainedTable = (Object[]) this.table[i + 1]; for (int j = 0; j < chainedTable.length; j += 2) { if (chainedTable[j] != null) { V value = (V) chainedTable[j + 1]; if (predicate.accept(value, parameter) == expected) { return onShortCircuit; } } } } else if (this.table[i] != null) { V value = (V) this.table[i + 1]; if (predicate.accept(value, parameter) == expected) { return onShortCircuit; } } } return atEnd; } @Override public boolean anySatisfy(Predicate predicate) { return this.shortCircuit(predicate, true, true, false); } @Override public

boolean anySatisfyWith(Predicate2 predicate, P parameter) { return this.shortCircuitWith(predicate, parameter, true, true, false); } @Override public boolean allSatisfy(Predicate predicate) { return this.shortCircuit(predicate, false, false, true); } @Override public

boolean allSatisfyWith(Predicate2 predicate, P parameter) { return this.shortCircuitWith(predicate, parameter, false, false, true); } @Override public boolean noneSatisfy(Predicate predicate) { return this.shortCircuit(predicate, true, false, true); } @Override public

boolean noneSatisfyWith(Predicate2 predicate, P parameter) { return this.shortCircuitWith(predicate, parameter, true, false, true); } protected class KeySet implements Set, Serializable, BatchIterable { private static final long serialVersionUID = 1L; @Override public boolean add(K key) { throw new UnsupportedOperationException("Cannot call add() on " + this.getClass().getSimpleName()); } @Override public boolean addAll(Collection collection) { throw new UnsupportedOperationException("Cannot call addAll() on " + this.getClass().getSimpleName()); } @Override public void clear() { UnifiedMap.this.clear(); } @Override public boolean contains(Object o) { return UnifiedMap.this.containsKey(o); } @Override public boolean containsAll(Collection collection) { for (Object aCollection : collection) { if (!UnifiedMap.this.containsKey(aCollection)) { return false; } } return true; } @Override public boolean isEmpty() { return UnifiedMap.this.isEmpty(); } @Override public Iterator iterator() { return new KeySetIterator(); } @Override public boolean remove(Object key) { int oldSize = UnifiedMap.this.occupied; UnifiedMap.this.remove(key); return UnifiedMap.this.occupied != oldSize; } @Override public boolean removeAll(Collection collection) { int oldSize = UnifiedMap.this.occupied; for (Object object : collection) { UnifiedMap.this.remove(object); } return oldSize != UnifiedMap.this.occupied; } public void putIfFound(Object key, Map other) { int index = UnifiedMap.this.index(key); Object cur = UnifiedMap.this.table[index]; if (cur != null) { Object val = UnifiedMap.this.table[index + 1]; if (cur == CHAINED_KEY) { this.putIfFoundFromChain((Object[]) val, (K) key, other); return; } if (UnifiedMap.this.nonNullTableObjectEquals(cur, (K) key)) { other.put(UnifiedMap.this.nonSentinel(cur), (V) val); } } } private void putIfFoundFromChain(Object[] chain, K key, Map other) { for (int i = 0; i < chain.length; i += 2) { Object k = chain[i]; if (k == null) { return; } if (UnifiedMap.this.nonNullTableObjectEquals(k, key)) { other.put(UnifiedMap.this.nonSentinel(k), (V) chain[i + 1]); } } } @Override public boolean retainAll(Collection collection) { int retainedSize = collection.size(); UnifiedMap retainedCopy = new UnifiedMap<>(retainedSize, UnifiedMap.this.loadFactor); for (Object key : collection) { this.putIfFound(key, retainedCopy); } if (retainedCopy.size() < this.size()) { UnifiedMap.this.maxSize = retainedCopy.maxSize; UnifiedMap.this.occupied = retainedCopy.occupied; UnifiedMap.this.table = retainedCopy.table; return true; } return false; } @Override public int size() { return UnifiedMap.this.size(); } @Override public void forEach(Procedure procedure) { UnifiedMap.this.forEachKey(procedure); } @Override public int getBatchCount(int batchSize) { return UnifiedMap.this.getBatchCount(batchSize); } @Override public void batchForEach(Procedure procedure, int sectionIndex, int sectionCount) { Object[] map = UnifiedMap.this.table; int sectionSize = map.length / sectionCount; int start = sectionIndex * sectionSize; int end = sectionIndex == sectionCount - 1 ? map.length : start + sectionSize; if (start % 2 != 0) { start++; } for (int i = start; i < end; i += 2) { Object cur = map[i]; if (cur == CHAINED_KEY) { UnifiedMap.this.chainedForEachKey((Object[]) map[i + 1], procedure); } else if (cur != null) { procedure.value(UnifiedMap.this.nonSentinel(cur)); } } } protected void copyKeys(Object[] result) { Object[] table = UnifiedMap.this.table; int count = 0; for (int i = 0; i < table.length; i += 2) { Object x = table[i]; if (x != null) { if (x == CHAINED_KEY) { Object[] chain = (Object[]) table[i + 1]; for (int j = 0; j < chain.length; j += 2) { Object cur = chain[j]; if (cur == null) { break; } result[count++] = UnifiedMap.this.nonSentinel(cur); } } else { result[count++] = UnifiedMap.this.nonSentinel(x); } } } } @Override public boolean equals(Object obj) { if (obj instanceof Set) { Set other = (Set) obj; if (other.size() == this.size()) { return this.containsAll(other); } } return false; } @Override public int hashCode() { int hashCode = 0; Object[] table = UnifiedMap.this.table; for (int i = 0; i < table.length; i += 2) { Object x = table[i]; if (x != null) { if (x == CHAINED_KEY) { Object[] chain = (Object[]) table[i + 1]; for (int j = 0; j < chain.length; j += 2) { Object cur = chain[j]; if (cur == null) { break; } hashCode += cur == NULL_KEY ? 0 : cur.hashCode(); } } else { hashCode += x == NULL_KEY ? 0 : x.hashCode(); } } } return hashCode; } @Override public String toString() { return Iterate.makeString(this, "[", ", ", "]"); } @Override public Object[] toArray() { int size = UnifiedMap.this.size(); Object[] result = new Object[size]; this.copyKeys(result); return result; } @Override public T[] toArray(T[] result) { int size = UnifiedMap.this.size(); if (result.length < size) { result = (T[]) Array.newInstance(result.getClass().getComponentType(), size); } this.copyKeys(result); if (size < result.length) { result[size] = null; } return result; } protected Object writeReplace() { UnifiedSet replace = UnifiedSet.newSet(UnifiedMap.this.size()); for (int i = 0; i < UnifiedMap.this.table.length; i += 2) { Object cur = UnifiedMap.this.table[i]; if (cur == CHAINED_KEY) { this.chainedAddToSet((Object[]) UnifiedMap.this.table[i + 1], replace); } else if (cur != null) { replace.add(UnifiedMap.this.nonSentinel(cur)); } } return replace; } private void chainedAddToSet(Object[] chain, UnifiedSet replace) { for (int i = 0; i < chain.length; i += 2) { Object cur = chain[i]; if (cur == null) { return; } replace.add(UnifiedMap.this.nonSentinel(cur)); } } } protected abstract class PositionalIterator implements Iterator { protected int count; protected int position; protected int chainPosition; protected boolean lastReturned; @Override public boolean hasNext() { return this.count < UnifiedMap.this.size(); } @Override public void remove() { if (!this.lastReturned) { throw new IllegalStateException("next() must be called as many times as remove()"); } this.count--; UnifiedMap.this.occupied--; if (this.chainPosition != 0) { this.removeFromChain(); return; } int pos = this.position - 2; Object cur = UnifiedMap.this.table[pos]; if (cur == CHAINED_KEY) { this.removeLastFromChain((Object[]) UnifiedMap.this.table[pos + 1], pos); return; } UnifiedMap.this.table[pos] = null; UnifiedMap.this.table[pos + 1] = null; this.position = pos; this.lastReturned = false; } protected void removeFromChain() { Object[] chain = (Object[]) UnifiedMap.this.table[this.position + 1]; int pos = this.chainPosition - 2; int replacePos = this.chainPosition; while (replacePos < chain.length - 2 && chain[replacePos + 2] != null) { replacePos += 2; } chain[pos] = chain[replacePos]; chain[pos + 1] = chain[replacePos + 1]; chain[replacePos] = null; chain[replacePos + 1] = null; this.chainPosition = pos; this.lastReturned = false; } protected void removeLastFromChain(Object[] chain, int tableIndex) { int pos = chain.length - 2; while (chain[pos] == null) { pos -= 2; } if (pos == 0) { UnifiedMap.this.table[tableIndex] = null; UnifiedMap.this.table[tableIndex + 1] = null; } else { chain[pos] = null; chain[pos + 1] = null; } this.lastReturned = false; } } protected class KeySetIterator extends PositionalIterator { protected K nextFromChain() { Object[] chain = (Object[]) UnifiedMap.this.table[this.position + 1]; Object cur = chain[this.chainPosition]; this.chainPosition += 2; if (this.chainPosition >= chain.length || chain[this.chainPosition] == null) { this.chainPosition = 0; this.position += 2; } this.lastReturned = true; return UnifiedMap.this.nonSentinel(cur); } @Override public K next() { if (!this.hasNext()) { throw new NoSuchElementException("next() called, but the iterator is exhausted"); } this.count++; Object[] table = UnifiedMap.this.table; if (this.chainPosition != 0) { return this.nextFromChain(); } while (table[this.position] == null) { this.position += 2; } Object cur = table[this.position]; if (cur == CHAINED_KEY) { return this.nextFromChain(); } this.position += 2; this.lastReturned = true; return UnifiedMap.this.nonSentinel(cur); } } private static boolean nullSafeEquals(Object value, Object other) { if (value == null) { if (other == null) { return true; } } else if (other == value || value.equals(other)) { return true; } return false; } protected class EntrySet implements Set>, Serializable, BatchIterable> { private static final long serialVersionUID = 1L; private transient WeakReference> holder = new WeakReference<>(UnifiedMap.this); @Override public boolean add(Entry entry) { throw new UnsupportedOperationException("Cannot call add() on " + this.getClass().getSimpleName()); } @Override public boolean addAll(Collection> collection) { throw new UnsupportedOperationException("Cannot call addAll() on " + this.getClass().getSimpleName()); } @Override public void clear() { UnifiedMap.this.clear(); } public boolean containsEntry(Entry entry) { return this.getEntry(entry) != null; } private Entry getEntry(Entry entry) { K key = (K) entry.getKey(); V value = (V) entry.getValue(); int index = UnifiedMap.this.index(key); Object cur = UnifiedMap.this.table[index]; Object curValue = UnifiedMap.this.table[index + 1]; if (cur == CHAINED_KEY) { return this.chainGetEntry((Object[]) curValue, key, value); } if (cur == null) { return null; } if (UnifiedMap.this.nonNullTableObjectEquals(cur, key)) { if (UnifiedMap.nullSafeEquals(value, curValue)) { return ImmutableEntry.of(UnifiedMap.this.nonSentinel(cur), (V) curValue); } } return null; } private Entry chainGetEntry(Object[] chain, K key, V value) { for (int i = 0; i < chain.length; i += 2) { Object cur = chain[i]; if (cur == null) { return null; } if (UnifiedMap.this.nonNullTableObjectEquals(cur, key)) { Object curValue = chain[i + 1]; if (UnifiedMap.nullSafeEquals(value, curValue)) { return ImmutableEntry.of(UnifiedMap.this.nonSentinel(cur), (V) curValue); } } } return null; } @Override public boolean contains(Object o) { return o instanceof Entry && this.containsEntry((Entry) o); } @Override public boolean containsAll(Collection collection) { for (Object obj : collection) { if (!this.contains(obj)) { return false; } } return true; } @Override public boolean isEmpty() { return UnifiedMap.this.isEmpty(); } @Override public Iterator> iterator() { return new EntrySetIterator(this.holder); } @Override public boolean remove(Object e) { if (!(e instanceof Entry)) { return false; } Entry entry = (Entry) e; K key = (K) entry.getKey(); V value = (V) entry.getValue(); int index = UnifiedMap.this.index(key); Object cur = UnifiedMap.this.table[index]; if (cur != null) { Object val = UnifiedMap.this.table[index + 1]; if (cur == CHAINED_KEY) { return this.removeFromChain((Object[]) val, key, value, index); } if (UnifiedMap.this.nonNullTableObjectEquals(cur, key) && UnifiedMap.nullSafeEquals(value, val)) { UnifiedMap.this.table[index] = null; UnifiedMap.this.table[index + 1] = null; UnifiedMap.this.occupied--; return true; } } return false; } private boolean removeFromChain(Object[] chain, K key, V value, int index) { for (int i = 0; i < chain.length; i += 2) { Object k = chain[i]; if (k == null) { return false; } if (UnifiedMap.this.nonNullTableObjectEquals(k, key)) { V val = (V) chain[i + 1]; if (UnifiedMap.nullSafeEquals(val, value)) { UnifiedMap.this.overwriteWithLastElementFromChain(chain, index, i); return true; } } } return false; } @Override public boolean removeAll(Collection collection) { boolean changed = false; for (Object obj : collection) { if (this.remove(obj)) { changed = true; } } return changed; } @Override public boolean retainAll(Collection collection) { int retainedSize = collection.size(); UnifiedMap retainedCopy = new UnifiedMap<>(retainedSize, UnifiedMap.this.loadFactor); for (Object obj : collection) { if (obj instanceof Entry) { Entry otherEntry = (Entry) obj; Entry thisEntry = this.getEntry(otherEntry); if (thisEntry != null) { retainedCopy.put(thisEntry.getKey(), thisEntry.getValue()); } } } if (retainedCopy.size() < this.size()) { UnifiedMap.this.maxSize = retainedCopy.maxSize; UnifiedMap.this.occupied = retainedCopy.occupied; UnifiedMap.this.table = retainedCopy.table; return true; } return false; } @Override public int size() { return UnifiedMap.this.size(); } @Override public void forEach(Procedure> procedure) { for (int i = 0; i < UnifiedMap.this.table.length; i += 2) { Object cur = UnifiedMap.this.table[i]; if (cur == CHAINED_KEY) { this.chainedForEachEntry((Object[]) UnifiedMap.this.table[i + 1], procedure); } else if (cur != null) { procedure.value(ImmutableEntry.of(UnifiedMap.this.nonSentinel(cur), (V) UnifiedMap.this.table[i + 1])); } } } private void chainedForEachEntry(Object[] chain, Procedure> procedure) { for (int i = 0; i < chain.length; i += 2) { Object cur = chain[i]; if (cur == null) { return; } procedure.value(ImmutableEntry.of(UnifiedMap.this.nonSentinel(cur), (V) chain[i + 1])); } } @Override public int getBatchCount(int batchSize) { return UnifiedMap.this.getBatchCount(batchSize); } @Override public void batchForEach(Procedure> procedure, int sectionIndex, int sectionCount) { Object[] map = UnifiedMap.this.table; int sectionSize = map.length / sectionCount; int start = sectionIndex * sectionSize; int end = sectionIndex == sectionCount - 1 ? map.length : start + sectionSize; if (start % 2 != 0) { start++; } for (int i = start; i < end; i += 2) { Object cur = map[i]; if (cur == CHAINED_KEY) { this.chainedForEachEntry((Object[]) map[i + 1], procedure); } else if (cur != null) { procedure.value(ImmutableEntry.of(UnifiedMap.this.nonSentinel(cur), (V) map[i + 1])); } } } protected void copyEntries(Object[] result) { Object[] table = UnifiedMap.this.table; int count = 0; for (int i = 0; i < table.length; i += 2) { Object x = table[i]; if (x != null) { if (x == CHAINED_KEY) { Object[] chain = (Object[]) table[i + 1]; for (int j = 0; j < chain.length; j += 2) { Object cur = chain[j]; if (cur == null) { break; } result[count++] = new WeakBoundEntry<>(UnifiedMap.this.nonSentinel(cur), (V) chain[j + 1], this.holder); } } else { result[count++] = new WeakBoundEntry<>(UnifiedMap.this.nonSentinel(x), (V) table[i + 1], this.holder); } } } } @Override public Object[] toArray() { Object[] result = new Object[UnifiedMap.this.size()]; this.copyEntries(result); return result; } @Override public T[] toArray(T[] result) { int size = UnifiedMap.this.size(); if (result.length < size) { result = (T[]) Array.newInstance(result.getClass().getComponentType(), size); } this.copyEntries(result); if (size < result.length) { result[size] = null; } return result; } private void readObject(ObjectInputStream in) throws IOException, ClassNotFoundException { in.defaultReadObject(); this.holder = new WeakReference<>(UnifiedMap.this); } @Override public boolean equals(Object obj) { if (obj instanceof Set) { Set other = (Set) obj; if (other.size() == this.size()) { return this.containsAll(other); } } return false; } @Override public int hashCode() { return UnifiedMap.this.hashCode(); } } protected class EntrySetIterator extends PositionalIterator> { private final WeakReference> holder; protected EntrySetIterator(WeakReference> holder) { this.holder = holder; } protected Entry nextFromChain() { Object[] chain = (Object[]) UnifiedMap.this.table[this.position + 1]; Object cur = chain[this.chainPosition]; Object value = chain[this.chainPosition + 1]; this.chainPosition += 2; if (this.chainPosition >= chain.length || chain[this.chainPosition] == null) { this.chainPosition = 0; this.position += 2; } this.lastReturned = true; return new WeakBoundEntry<>(UnifiedMap.this.nonSentinel(cur), (V) value, this.holder); } @Override public Entry next() { if (!this.hasNext()) { throw new NoSuchElementException("next() called, but the iterator is exhausted"); } this.count++; Object[] table = UnifiedMap.this.table; if (this.chainPosition != 0) { return this.nextFromChain(); } while (table[this.position] == null) { this.position += 2; } Object cur = table[this.position]; Object value = table[this.position + 1]; if (cur == CHAINED_KEY) { return this.nextFromChain(); } this.position += 2; this.lastReturned = true; return new WeakBoundEntry<>(UnifiedMap.this.nonSentinel(cur), (V) value, this.holder); } } protected static class WeakBoundEntry implements Map.Entry { protected final K key; protected V value; protected final WeakReference> holder; protected WeakBoundEntry(K key, V value, WeakReference> holder) { this.key = key; this.value = value; this.holder = holder; } @Override public K getKey() { return this.key; } @Override public V getValue() { return this.value; } @Override public V setValue(V value) { this.value = value; UnifiedMap map = this.holder.get(); if (map != null && map.containsKey(this.key)) { return map.put(this.key, value); } return null; } @Override public boolean equals(Object obj) { if (obj instanceof Entry) { Entry other = (Entry) obj; K otherKey = (K) other.getKey(); V otherValue = (V) other.getValue(); return UnifiedMap.nullSafeEquals(this.key, otherKey) && UnifiedMap.nullSafeEquals(this.value, otherValue); } return false; } @Override public int hashCode() { return (this.key == null ? 0 : this.key.hashCode()) ^ (this.value == null ? 0 : this.value.hashCode()); } @Override public String toString() { return this.key + "=" + this.value; } } protected class ValuesCollection extends ValuesCollectionCommon implements Serializable, BatchIterable { private static final long serialVersionUID = 1L; @Override public void clear() { UnifiedMap.this.clear(); } @Override public boolean contains(Object o) { return UnifiedMap.this.containsValue(o); } @Override public boolean containsAll(Collection collection) { // todo: this is N^2. if c is large, we should copy the values to a set. return Iterate.allSatisfy(collection, Predicates.in(this)); } @Override public boolean isEmpty() { return UnifiedMap.this.isEmpty(); } @Override public Iterator iterator() { return new ValuesIterator(); } @Override public boolean remove(Object o) { // this is so slow that the extra overhead of the iterator won't be noticeable if (o == null) { for (Iterator it = this.iterator(); it.hasNext(); ) { if (it.next() == null) { it.remove(); return true; } } } else { for (Iterator it = this.iterator(); it.hasNext(); ) { V o2 = it.next(); if (o == o2 || o2.equals(o)) { it.remove(); return true; } } } return false; } @Override public boolean removeAll(Collection collection) { // todo: this is N^2. if c is large, we should copy the values to a set. boolean changed = false; for (Object obj : collection) { if (this.remove(obj)) { changed = true; } } return changed; } @Override public boolean retainAll(Collection collection) { boolean modified = false; Iterator e = this.iterator(); while (e.hasNext()) { if (!collection.contains(e.next())) { e.remove(); modified = true; } } return modified; } @Override public int size() { return UnifiedMap.this.size(); } @Override public void forEach(Procedure procedure) { UnifiedMap.this.forEachValue(procedure); } @Override public int getBatchCount(int batchSize) { return UnifiedMap.this.getBatchCount(batchSize); } @Override public void batchForEach(Procedure procedure, int sectionIndex, int sectionCount) { UnifiedMap.this.batchForEach(procedure, sectionIndex, sectionCount); } protected void copyValues(Object[] result) { int count = 0; for (int i = 0; i < UnifiedMap.this.table.length; i += 2) { Object x = UnifiedMap.this.table[i]; if (x != null) { if (x == CHAINED_KEY) { Object[] chain = (Object[]) UnifiedMap.this.table[i + 1]; for (int j = 0; j < chain.length; j += 2) { Object cur = chain[j]; if (cur == null) { break; } result[count++] = chain[j + 1]; } } else { result[count++] = UnifiedMap.this.table[i + 1]; } } } } @Override public Object[] toArray() { int size = UnifiedMap.this.size(); Object[] result = new Object[size]; this.copyValues(result); return result; } @Override public T[] toArray(T[] result) { int size = UnifiedMap.this.size(); if (result.length < size) { result = (T[]) Array.newInstance(result.getClass().getComponentType(), size); } this.copyValues(result); if (size < result.length) { result[size] = null; } return result; } protected Object writeReplace() { FastList replace = FastList.newList(UnifiedMap.this.size()); for (int i = 0; i < UnifiedMap.this.table.length; i += 2) { Object cur = UnifiedMap.this.table[i]; if (cur == CHAINED_KEY) { this.chainedAddToList((Object[]) UnifiedMap.this.table[i + 1], replace); } else if (cur != null) { replace.add((V) UnifiedMap.this.table[i + 1]); } } return replace; } private void chainedAddToList(Object[] chain, FastList replace) { for (int i = 0; i < chain.length; i += 2) { Object cur = chain[i]; if (cur == null) { return; } replace.add((V) chain[i + 1]); } } @Override public String toString() { return Iterate.makeString(this, "[", ", ", "]"); } } protected class ValuesIterator extends PositionalIterator { protected V nextFromChain() { Object[] chain = (Object[]) UnifiedMap.this.table[this.position + 1]; V val = (V) chain[this.chainPosition + 1]; this.chainPosition += 2; if (this.chainPosition >= chain.length || chain[this.chainPosition] == null) { this.chainPosition = 0; this.position += 2; } this.lastReturned = true; return val; } @Override public V next() { if (!this.hasNext()) { throw new NoSuchElementException("next() called, but the iterator is exhausted"); } this.count++; Object[] table = UnifiedMap.this.table; if (this.chainPosition != 0) { return this.nextFromChain(); } while (table[this.position] == null) { this.position += 2; } Object cur = table[this.position]; Object val = table[this.position + 1]; if (cur == CHAINED_KEY) { return this.nextFromChain(); } this.position += 2; this.lastReturned = true; return (V) val; } } private K nonSentinel(Object key) { return key == NULL_KEY ? null : (K) key; } private static Object toSentinelIfNull(Object key) { if (key == null) { return NULL_KEY; } return key; } private boolean nonNullTableObjectEquals(Object cur, K key) { return cur == key || (cur == NULL_KEY ? key == null : cur.equals(key)); } @Override public ImmutableMap toImmutable() { return Maps.immutable.withAll(this); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy