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

com.hazelcast.util.collection.Long2LongHashMap Maven / Gradle / Ivy

/*
 * Original work Copyright 2015 Real Logic Ltd.
 * Modified work Copyright (c) 2015, Hazelcast, Inc. All Rights Reserved.
 *
 * 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.hazelcast.util.collection;

import com.hazelcast.util.QuickMath;
import com.hazelcast.util.function.BiConsumer;
import com.hazelcast.util.function.LongLongConsumer;
import com.hazelcast.util.function.Predicate;
import com.hazelcast.util.function.Supplier;
import edu.umd.cs.findbugs.annotations.SuppressFBWarnings;

import java.util.Arrays;
import java.util.Collection;
import java.util.Iterator;
import java.util.Map;
import java.util.NoSuchElementException;
import java.util.Set;

import static com.hazelcast.util.collection.Hashing.evenLongHash;

/**
 * A Probing hashmap specialised for long key and value pairs.
 */
public class Long2LongHashMap implements Map {
    /** The default load factor for constructors not explicitly supplying it */
    public static final double DEFAULT_LOAD_FACTOR = 0.6;
    private final Set keySet;
    private final LongIterator valueIterator;
    private final Collection values;
    private final Set> entrySet;

    private final double loadFactor;
    private final long missingValue;

    private long[] entries;
    private int capacity;
    private int mask;
    private int resizeThreshold;
    private int size;

    public Long2LongHashMap(int initialCapacity, double loadFactor, long missingValue) {
        this(loadFactor, missingValue);
        capacity(QuickMath.nextPowerOfTwo(initialCapacity));
    }

    public Long2LongHashMap(long missingValue) {
        this(16, DEFAULT_LOAD_FACTOR, missingValue);
    }

    public Long2LongHashMap(Long2LongHashMap that) {
        this(that.loadFactor, that.missingValue);
        this.entries = Arrays.copyOf(that.entries, that.entries.length);
        this.capacity = that.capacity;
        this.mask = that.mask;
        this.resizeThreshold = that.resizeThreshold;
        this.size = that.size;
    }

    private Long2LongHashMap(double loadFactor, long missingValue) {
        this.entrySet = entrySetSingleton();
        this.keySet = keySetSingleton();
        this.values = valuesSingleton();
        this.valueIterator = new LongIterator(1);
        this.loadFactor = loadFactor;
        this.missingValue = missingValue;
    }

    /**
     * {@inheritDoc}
     */
    public int size() {
        return size;
    }

    /**
     * {@inheritDoc}
     */
    public boolean isEmpty() {
        return size() == 0;
    }

    public long get(final long key) {
        final long[] entries = this.entries;
        int index = evenLongHash(key, mask);
        long candidateKey;
        while ((candidateKey = entries[index]) != missingValue) {
            if (candidateKey == key) {
                return entries[index + 1];
            }
            index = next(index);
        }
        return missingValue;
    }

    public long put(final long key, final long value) {
        assert key != missingValue : "Invalid key " + key;
        assert value != missingValue : "Invalid value " + value;
        long oldValue = missingValue;
        int index = evenLongHash(key, mask);
        long candidateKey;
        while ((candidateKey = entries[index]) != missingValue) {
            if (candidateKey == key) {
                oldValue = entries[index + 1];
                break;
            }
            index = next(index);
        }
        if (oldValue == missingValue) {
            ++size;
            entries[index] = key;
        }
        entries[index + 1] = value;
        checkResize();
        return oldValue;
    }

    private void checkResize() {
        if (size > resizeThreshold) {
            final int newCapacity = capacity << 1;
            if (newCapacity < 0) {
                throw new IllegalStateException("Max capacity reached at size=" + size);
            }
            rehash(newCapacity);
        }
    }

    private void rehash(final int newCapacity) {
        final long[] oldEntries = entries;
        capacity(newCapacity);
        for (int i = 0; i < oldEntries.length; i += 2) {
            final long key = oldEntries[i];
            if (key != missingValue) {
                put(key, oldEntries[i + 1]);
            }
        }
    }

    /**
     * Primitive specialised forEach implementation.
     * 

* NB: Renamed from forEach to avoid overloading on parameter types of lambda * expression, which doesn't interplay well with type inference in lambda expressions. * * @param consumer a callback called for each key/value pair in the map. */ public void longForEach(final LongLongConsumer consumer) { final long[] entries = this.entries; for (int i = 0; i < entries.length; i += 2) { final long key = entries[i]; if (key != missingValue) { consumer.accept(entries[i], entries[i + 1]); } } } /** * Provides a cursor over the map's entries. Similar to {@code entrySet().iterator()}, * but with a simpler and more clear API. */ public LongLongCursor cursor() { return new LongLongCursor(); } /** * Implements the cursor. */ public final class LongLongCursor { private int i = -2; public boolean advance() { final long[] es = entries; do { i += 2; } while (i < es.length && es[i] == missingValue); return i < es.length; } public long key() { return entries[i]; } public long value() { return entries[i + 1]; } } /** * Long primitive specialised containsKey. * * @param key the key to check. * @return true if the map contains key as a key, false otherwise. */ public boolean containsKey(final long key) { return get(key) != missingValue; } public boolean containsValue(final long value) { final long[] entries = this.entries; for (int i = 1; i < entries.length; i += 2) { final long entryValue = entries[i]; if (entryValue == value) { return true; } } return false; } /** * {@inheritDoc} */ public void clear() { Arrays.fill(entries, missingValue); size = 0; } // ---------------- Boxed Versions Below ---------------- /** * {@inheritDoc} */ public Long get(final Object key) { return get((long) (Long) key); } /** * {@inheritDoc} */ public Long put(final Long key, final Long value) { return put(key.longValue(), value.longValue()); } /** * {@inheritDoc} */ public void forEach(final BiConsumer action) { longForEach(new UnboxingBiConsumer(action)); } /** * {@inheritDoc} */ public boolean containsKey(final Object key) { return containsKey((long) (Long) key); } /** * {@inheritDoc} */ public boolean containsValue(final Object value) { return containsValue((long) (Long) value); } /** * {@inheritDoc} */ public void putAll(final Map map) { for (final Entry entry : map.entrySet()) { put(entry.getKey(), entry.getValue()); } } /** * {@inheritDoc} */ public Set keySet() { return keySet; } /** * {@inheritDoc} */ public Collection values() { return values; } /** * {@inheritDoc} * This set's iterator also implements Map.Entry * so the next() method can just return the iterator * instance itself with no heap allocation. This characteristic * makes the set unusable wherever the returned entries are * retained (such as coll.addAll(entrySet). */ public Set> entrySet() { return entrySet; } /** * {@inheritDoc} */ public Long remove(final Object key) { return remove((long) (Long) key); } public long remove(final long key) { final long[] entries = this.entries; int index = evenLongHash(key, mask); long candidateKey; while ((candidateKey = entries[index]) != missingValue) { if (candidateKey == key) { final int valueIndex = index + 1; final long oldValue = entries[valueIndex]; entries[index] = missingValue; entries[valueIndex] = missingValue; size--; compactChain(index); return oldValue; } index = next(index); } return missingValue; } @Override public String toString() { final StringBuilder b = new StringBuilder(size() * 8); b.append('{'); longForEach(new LongLongConsumer() { String separator = ""; @Override public void accept(long key, long value) { b.append(separator).append(key).append("->").append(value); separator = " "; } }); return b.append('}').toString(); } private void compactChain(int deleteIndex) { final long[] entries = this.entries; int index = deleteIndex; while (true) { index = next(index); if (entries[index] == missingValue) { return; } final int hash = evenLongHash(entries[index], mask); if ((index < hash && (hash <= deleteIndex || deleteIndex <= index)) || (hash <= deleteIndex && deleteIndex <= index)) { entries[deleteIndex] = entries[index]; entries[deleteIndex + 1] = entries[index + 1]; entries[index] = missingValue; entries[index + 1] = missingValue; deleteIndex = index; } } } private static class IteratorSupplier implements Supplier> { private final LongIterator keyIterator; public IteratorSupplier(LongIterator keyIterator) { this.keyIterator = keyIterator; } @Override public Iterator get() { return keyIterator.reset(); } } private static class EntryIteratorSupplier implements Supplier>> { private final EntryIterator entryIterator; public EntryIteratorSupplier(EntryIterator entryIterator) { this.entryIterator = entryIterator; } @Override public Iterator> get() { return entryIterator.reset(); } } private static class UnboxingBiConsumer implements LongLongConsumer { private final BiConsumer action; public UnboxingBiConsumer(BiConsumer action) { this.action = action; } @Override public void accept(long t, long u) { action.accept(t, u); } } // ---------------- Utility Classes ---------------- private abstract class AbstractIterator { private int capacity; private int mask; private int positionCounter; private int stopCounter; private void reset() { final long[] entries = Long2LongHashMap.this.entries; capacity = entries.length; mask = capacity - 1; int i = capacity; if (entries[capacity - 2] != missingValue) { i = 0; for (int size = capacity; i < size; i += 2) { if (entries[i] == missingValue) { break; } } } stopCounter = i; positionCounter = i + capacity; } protected int keyPosition() { return positionCounter & mask; } public boolean hasNext() { final long[] entries = Long2LongHashMap.this.entries; boolean hasNext = false; for (int i = positionCounter - 2; i >= stopCounter; i -= 2) { final int index = i & mask; if (entries[index] != missingValue) { hasNext = true; break; } } return hasNext; } protected void findNext() { final long[] entries = Long2LongHashMap.this.entries; for (int i = positionCounter - 2; i >= stopCounter; i -= 2) { final int index = i & mask; if (entries[index] != missingValue) { positionCounter = i; return; } } throw new NoSuchElementException(); } public void remove() { throw new UnsupportedOperationException("remove"); } } private final class LongIterator extends AbstractIterator implements Iterator { private final int offset; private LongIterator(final int offset) { this.offset = offset; } public Long next() { return nextValue(); } public long nextValue() { findNext(); return entries[keyPosition() + offset]; } public LongIterator reset() { super.reset(); return this; } } @SuppressFBWarnings(value = "PZ_DONT_REUSE_ENTRY_OBJECTS_IN_ITERATORS", justification = "deliberate, documented choice") private final class EntryIterator extends AbstractIterator implements Iterator>, Entry { private long key; private long value; private EntryIterator() { } public Long getKey() { return key; } public Long getValue() { return value; } public Long setValue(final Long value) { throw new UnsupportedOperationException(); } public Entry next() { findNext(); final int keyPosition = keyPosition(); key = entries[keyPosition]; value = entries[keyPosition + 1]; return this; } public EntryIterator reset() { super.reset(); key = missingValue; value = missingValue; return this; } } private int next(final int index) { return (index + 2) & mask; } private void capacity(final int newCapacity) { capacity = newCapacity; resizeThreshold = (int) (newCapacity * loadFactor); mask = (newCapacity * 2) - 1; entries = new long[newCapacity * 2]; size = 0; Arrays.fill(entries, missingValue); } private MapDelegatingSet> entrySetSingleton() { return new MapDelegatingSet>(this, new EntryIteratorSupplier(new EntryIterator()), new Predicate() { @SuppressWarnings("unchecked") @Override public boolean test(Object e) { return containsKey(((Entry) e).getKey()); } }); } private MapDelegatingSet keySetSingleton() { return new MapDelegatingSet(this, new IteratorSupplier(new LongIterator(0)), new Predicate() { @Override public boolean test(Object value) { return containsValue(value); } }); } private MapDelegatingSet valuesSingleton() { return new MapDelegatingSet(this, new Supplier>() { @Override public Iterator get() { return valueIterator.reset(); } }, new Predicate() { @Override public boolean test(Object key) { return containsKey(key); } }); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy