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

org.apache.ignite.internal.util.collection.IntHashMap Maven / Gradle / Ivy

Go to download

Java-based middleware for in-memory processing of big data in a distributed environment.

There is a newer version: 3.0.0-beta1
Show newest version
/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You 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 org.apache.ignite.internal.util.collection;

import java.util.Arrays;
import java.util.Objects;
import java.util.stream.Collectors;

/**
 * The simple map for primitive types base on Robin-hood hashing with backward shift.
 */
public class IntHashMap implements IntMap {
    /** Initial capacity. */
    public static final int INITIAL_CAPACITY = 8;

    /** Maximum capacity. */
    public static final int MAXIMUM_CAPACITY = 1 << 30;

    /** Magic hash mixer. */
    private static final int MAGIC_HASH_MIXER = 0x9E3779B9;

    /** Array load percentage before resize. */
    private static final float SCALE_LOAD_FACTOR = 0.7F;

    /** Array load percentage before decrise size. */
    private static final float COMPACT_LOAD_FACTOR = 0.2F;

    /** Scale threshold. */
    private int scaleThreshold;

    /** Compact threshold. */
    private int compactThreshold;

    /** Entries. */
    private Entry[] entries;

    /** Count of elements in Map. */
    private int size;

    /** Inner entry structure. */
    private static class Entry {
        /** Key. */
        public final int key;

        /** Value. */
        public final V val;

        /** Default constructor. */
        Entry(int key, V val) {
            this.key = key;
            this.val = val;
        }

        /** {@inheritDoc} */
        @Override public String toString() {
            return "{key=" + key + ", val=" + val + '}';
        }
    }

    /**
     * Returns required size of table.
     *
     * @param cap Capacity.
     */
    static int tableSize(int cap) {
        int n = cap - 1;
        n |= n >>> 1;
        n |= n >>> 2;
        n |= n >>> 4;
        n |= n >>> 8;
        n |= n >>> 16;

        return  (n < INITIAL_CAPACITY) ? INITIAL_CAPACITY : (n >= MAXIMUM_CAPACITY) ? MAXIMUM_CAPACITY : n + 1;
    }

    /** Default constructor. */
    public IntHashMap() {
        entries = (Entry[])new Entry[INITIAL_CAPACITY];
    }

    /** Create map with preallocated array. */
    public IntHashMap(int cap) {
        int entriesSize = tableSize(cap);

        compactThreshold = (int)(COMPACT_LOAD_FACTOR * (entriesSize >> 1));

        scaleThreshold = (int)(entriesSize * SCALE_LOAD_FACTOR);

        entries = (Entry[])new Entry[entriesSize];
    }

    /** {@inheritDoc} */
    @Override public V get(int key) {
        int idx = find(key);

        return idx < 0 ? null : entries[idx].val;
    }

    /** {@inheritDoc} */
    @Override public V put(int key, V val) {
        return put0(new Entry<>(key, val));
    }

    /** {@inheritDoc} */
    @Override public V remove(int key) {
        if (entries.length != INITIAL_CAPACITY && compactThreshold > size)
            resize(false);

        int idx = find(key);

        if (idx < 0)
            return null;

        size--;

        V tmp = entries[idx].val;

        // Backward shift
        for (int i = 0; i < entries.length; i++) {
            int curIdx = (idx + i) & (entries.length - 1);
            int nextIdx = (idx + i + 1) & (entries.length - 1);

            Entry nextEntry = entries[nextIdx];

            if (nextEntry == null || distance(nextIdx, nextEntry.key) == 0) {
                entries[curIdx] = null;

                return tmp;
            }

            entries[curIdx] = nextEntry;
        }

        throw new IllegalStateException("Unreachable state exception. Backward shift has a problem. " +
            "Removing key: " + key + " map state: " + toString());
    }

    /** {@inheritDoc} */
    @Override public V putIfAbsent(int key, V val) {
        int idx = find(key);

        if (idx < 0) {
            put(key, val);

            return null;
        }
        else
            return entries[idx].val;
    }

    /** {@inheritDoc} */
    @Override public  void forEach(EntryConsumer act) throws E {
        for (Entry entry : entries)
            if (entry != null)
                act.accept(entry.key, entry.val);
    }

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

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

    /** {@inheritDoc} */
    @Override public boolean containsKey(int key) {
        return find(key) >= 0;
    }

    /** {@inheritDoc} */
    @Override public boolean containsValue(V val) {
        return Arrays.stream(entries)
            .filter(Objects::nonNull)
            .anyMatch(entry -> Objects.equals(val, entry.val));
    }

    /** {@inheritDoc} */
    @Override public String toString() {
        String strEntries = Arrays.stream(entries)
            .filter(Objects::nonNull)
            .map(Entry::toString)
            .collect(Collectors.joining(","));

        return "IntHashMap{size=" + size + ", entries=[" + strEntries + "]}";
    }

    /**
     * Returns size of shift from ideal position computing via index(key).
     *
     * @param curIdx Current index.
     * @param key Target key.
     */
    protected int distance(int curIdx, int key) {
        int keyIdx = index(key);

        return curIdx >= keyIdx ? curIdx - keyIdx : entries.length - keyIdx + curIdx;
    }

    /**
     * Position in entites array.
     *
     * @param key Targert key.
     */
    protected int index(int key) {
        return (entries.length - 1) & ((key ^ (key >>> 16)) * MAGIC_HASH_MIXER);
    }

    /**
     * Does insert of entry.
     */
    private V put0(Entry entry) {
        if (size >= scaleThreshold)
            resize(true);

        Entry savedEntry = entry;

        int startKey = savedEntry.key;

        for (int i = 0; i < entries.length; i++) {
            int idx = (index(startKey) + i) & (entries.length - 1);

            Entry curEntry = entries[idx];

            if (curEntry == null) {
                entries[idx] = savedEntry;

                size++;

                return null;
            }
            else if (curEntry.key == savedEntry.key) {
                entries[idx] = savedEntry;

                return curEntry.val;
            }

            int curDist = distance(idx, curEntry.key);
            int savedDist = distance(idx, savedEntry.key);

            if (curDist < savedDist) {
                entries[idx] = savedEntry;

                savedEntry = curEntry;
            }
        }

        throw new IllegalStateException("Unreachable state exception. Insertion position not found. " +
            "Entry: " + entry + " map state: " + toString());
    }

    /**
     * @param key Key.
     * @return position of element or -1 if absent.
     */
    private int find(int key) {
        int idx = index(key);

        for (int keyDist = 0; keyDist < entries.length; keyDist++) {
            int curIdx = (idx + keyDist) & (entries.length - 1);

            Entry entry = entries[curIdx];

            if (entry == null)
                return -1;
            else if (entry.key == key)
                return curIdx;

            int entryDist = distance(curIdx, entry.key);

            if (keyDist > entryDist)
                return -1;
        }

        return -1;
    }

    /**
     * Does resize of entities array.
     */
    private void resize(boolean increase) {
        if (MAXIMUM_CAPACITY == entries.length)
            throw new IllegalStateException("Maximum capacity: " + MAXIMUM_CAPACITY + " is reached.");

        Entry[] oldEntries = entries;

        int newCap = increase ? entries.length << 1 : entries.length >> 1;

        entries = (Entry[])new Entry[newCap];

        compactThreshold = (int)(COMPACT_LOAD_FACTOR * (entries.length >> 1));

        scaleThreshold = (int)(entries.length * SCALE_LOAD_FACTOR);

        size = 0;

        for (Entry entry : oldEntries)
            if (entry != null)
                put0(entry);
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy