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

com.koloboke.collect.impl.hash.MutableDHash Maven / Gradle / Ivy

/* with
 DHash|QHash hash
 Mutable mutability
 true|false concurrentModificationChecked
*/
/*
 * Copyright 2014 the original author or authors.
 *
 * 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.koloboke.collect.impl.hash;

import com.koloboke.collect.hash.HashConfig;
import com.koloboke.collect.hash.HashOverflowException;
import com.koloboke.collect.impl.AbstractContainer;

import javax.annotation.Nonnull;

import static com.koloboke.collect.impl.hash.DHashCapacities.nearestGreaterCapacity;


public abstract class MutableDHash extends AbstractContainer implements DHash {

    private static int minFreeSlots(int capacity, int size, double maxLoad, int maxSize) {
        double load = (double) size / (double) capacity;
        // See "Tombstones purge from hashtable: theory and practice" wiki page
        double rehashLoad =
                /* if DHash hash */  0.49 + 0.866 * load - 0.363
                /* elif QHash hash //0.55 + 0.721 * load - 0.274// endif */ * load * load;

        int minFreeSlots;
        // minFreeSlots shouldn't be less than `capacity - maxSize`
        if (rehashLoad > maxLoad) {
            minFreeSlots = (int) ((double) capacity * (1.0 - rehashLoad));
        } else {
            minFreeSlots = capacity - maxSize;
        }
        // Need at least one free slot for open addressing
        return minFreeSlots > 0 ? minFreeSlots : 1;
    }

    ////////////////////////////
    // Fields

    private HashConfigWrapper configWrapper;


    /** The current number of occupied slots in the hash. */
    int size;

    private int maxSize;

    /** The current number of free slots in the hash. */
    private int freeSlots;


    private int minFreeSlots;


    private int removedSlots;


    /* if true concurrentModificationChecked */ private int modCount = 0;/* endif */


    /////////////////////////////
    // Getters

    @Nonnull
    @Override
    public final HashConfig hashConfig() {
        return configWrapper.config();
    }

    @Override
    public final HashConfigWrapper configWrapper() {
        return configWrapper;
    }

    @Override
    public final int size() {
        return size;
    }

    @Override
    public abstract int capacity();

    @Override
    public final boolean noRemoved() {
        return removedSlots == 0;
    }

    @Override
    public final int freeSlots() {
        return freeSlots;
    }

    @Override
    public final int removedSlots() {
        return removedSlots;
    }

    @Override
    public final int modCount() {
        /* if true concurrentModificationChecked */
        return modCount;
        /* elif false concurrentModificationChecked //
        return 0;
        // endif */
    }

    /* if true concurrentModificationChecked */
    final void incrementModCount() {
        modCount++;
    }
    /* endif */

    /** For tests */
    public final int maxSize() {
        return maxSize;
    }

    @Override
    public final double currentLoad() {
        return ((double) (size + removedSlots)) / (double) capacity();
    }


    ////////////////////////
    // Initialization and construction operations

    /**
     * Root operation for copy constructors
     *
     * @param hash Mutable or Immutable DHash instance
     */
    final void copy(DHash hash) {
        this.configWrapper = hash.configWrapper();
        int size = this.size = hash.size();
        int capacity = hash.capacity();
        this.maxSize = maxSize(capacity);
        int freeSlots = this.freeSlots = hash.freeSlots();
        int minFreeSlots = this.minFreeSlots =
                minFreeSlots(capacity, size, hashConfig().getMaxLoad(), maxSize);
        // see #initSlotCounts()
        if (freeSlots < minFreeSlots) this.minFreeSlots = (freeSlots + 1) / 2;
        this.removedSlots = hash.removedSlots();
    }

    /**
     * Creates data structures with a prime capacity at or near the minimum
     * needed to hold {@code size} elements without triggering a rehash.
     *
     * 

Should be called only in constructors and externalization code. */ final void init(HashConfigWrapper configWrapper, int size) { this.configWrapper = configWrapper; this.size = 0; internalInit(targetCapacity(size)); } private void internalInit(int capacity) { initSlotCounts(capacity); allocateArrays(capacity); } /** * Allocates arrays of {@code capacity} size to hold states, elements, keys * or values in. * *

Subclasses should override this method, but SHOULD NOT call it. This * method is called in MutableDHash from initForRehash() and init() methods. * * @param capacity size of arrays, comprising the hash */ abstract void allocateArrays(int capacity); private void initSlotCounts(int capacity) { maxSize = maxSize(capacity); minFreeSlots = minFreeSlots(capacity, size, hashConfig().getMaxLoad(), maxSize); int freeSlots = this.freeSlots = capacity - size; // free could be less than minFreeSlots only in case when capacity // is not sufficient to comply load factor (due to saturation with // Java array size limit). Set minFreeSlots to a half of free to avoid // too often (instant) rehashing in this case. if (freeSlots < minFreeSlots) this.minFreeSlots = (freeSlots + 1) / 2; removedSlots = 0; } private int maxSize(int capacity) { // No sense in trying to rehash after each insertion // if the capacity is already reached the limit. return !isMaxCapacity(capacity) ? configWrapper.maxSize(capacity) : capacity - 1; } /** * Moves elements to the new arrays of {@code newCapacity} size. * *

This method should be implemented as follows: * * 1. Copy references to the old arrays comprising the hash from fields * to local variables * * 2. Call {@link #initForRehash(int)} * * 3. Move elements, entries, etc. from the old arrays to the new ones. * *

Subclasses should implement, but MUST NOT call this method. This method is called * in DHash from postInsertHooks, {@link #ensureCapacity(long)} and {@link #shrink()} methods. */ abstract void rehash(int newCapacity); /** * This method just increments modification count (see {@link #modCount()}) * and calls {@link #internalInit(int)}. Should be called by subclasses in * {@link #rehash(int)} implementation. */ final void initForRehash(int newCapacity) { /* if true concurrentModificationChecked */modCount++;/* endif */ internalInit(newCapacity); } ////////////////////////////// // Roots of chain operations /** * Empties the hash. */ @Override public void clear() { /* if true concurrentModificationChecked */modCount++;/* endif */ size = 0; freeSlots = capacity(); removedSlots = 0; } abstract void removeAt(int index); ///////////////////////////// // Modification hooks and rehash logic @Override public boolean shrink() { int newCapacity = targetCapacity(size); if (removedSlots > 0 || newCapacity < capacity()) { rehash(newCapacity); return true; } else { return false; } } private boolean tryRehashForExpansion(int newCapacity) { // No sense in rehashing for expansion if we already reached Java array // size limit. if (newCapacity > capacity() || removedSlots > 0) { rehash(newCapacity); return true; } else { if (freeSlots < minFreeSlots) minFreeSlots = (freeSlots + 1) / 2; return false; } } @Override public final boolean ensureCapacity(long minSize) { int intMinSize = (int) Math.min(minSize, (long) Integer.MAX_VALUE); if (minSize < 0L) throw new IllegalArgumentException( "Min size should be positive, " + minSize + " given."); int additionalSize = intMinSize - size; if (additionalSize <= 0) return false; if (intMinSize > maxSize || freeSlots - additionalSize < minFreeSlots) { return tryRehashForExpansion(targetCapacity(intMinSize)); } else { return false; } } final void postRemoveHook() { size--; removedSlots++; } final void postFreeSlotInsertHook() { if (++size > maxSize) { if (tryRehashForExpansion(grownCapacity())) return; } if (--freeSlots < minFreeSlots) { if (!tryRehashIfTooFewFreeSlots() && freeSlots == 0) { throw new HashOverflowException(); } } } final void postRemovedSlotInsertHook() { if (++size > maxSize) { if (tryRehashForExpansion(grownCapacity())) return; } removedSlots--; } private boolean tryRehashIfTooFewFreeSlots() { if (removedSlots > 0) { rehash(targetCapacity(size)); return true; } else { return tryRehashForExpansion(grownCapacity()); } } /** @see MutableLHash#doubleSizedArrays() */ boolean doubleSizedArrays() { return false; } private int targetCapacity(int size) { return DHashCapacities.capacity(configWrapper, size, doubleSizedArrays()); } private boolean isMaxCapacity(int capacity) { return DHashCapacities.isMaxCapacity(capacity, doubleSizedArrays()); } private int grownCapacity() { return nearestGreaterCapacity(configWrapper.grow(capacity()), size, doubleSizedArrays()); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy