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

com.annimon.stream.internal.SpinedBuffer Maven / Gradle / Ivy

package com.annimon.stream.internal;

import java.util.Arrays;
import java.util.Iterator;

import com.annimon.stream.function.DoubleConsumer;
import com.annimon.stream.function.IntConsumer;
import com.annimon.stream.function.LongConsumer;
import com.annimon.stream.iterator.PrimitiveIterator;

final class SpinedBuffer {

    /**
     * Minimum power-of-two for the first chunk.
     */
    static final int MIN_CHUNK_POWER = 4;

    /**
     * Minimum size for the first chunk.
     */
    static final int MIN_CHUNK_SIZE = 1 << MIN_CHUNK_POWER;

    /**
     * Max power-of-two for chunks.
     */
    private static final int MAX_CHUNK_POWER = 30;

    /**
     * Minimum array size for array-of-chunks.
     */
    static final int MIN_SPINE_SIZE = 8;

    private SpinedBuffer() {
    }

    /**
     * Base class for a data structure for gathering elements into a buffer and then
     * iterating them. Maintains an array of increasingly sized arrays, so there is
     * no copying cost associated with growing the data structure.
     */
    abstract static class OfPrimitive implements Iterable {

        /**
         * log2 of the size of the first chunk.
         */
        final int initialChunkPower;

        /**
         * Index of the *next* element to write; may point into, or just outside of,
         * the current chunk.
         */
        int elementIndex;

        /**
         * Index of the *current* chunk in the spine array, if the spine array is
         * non-null.
         */
        int spineIndex;

        /**
         * Count of elements in all prior chunks.
         */
        long[] priorElementCount;

        T_ARR curChunk;

        T_ARR[] spine;

        /**
         * Construct with a specified initial capacity.
         *
         * @param initialCapacity The minimum expected number of elements
         */
        OfPrimitive(int initialCapacity) {
            if (initialCapacity < 0)
                throw new IllegalArgumentException("Illegal Capacity: " + initialCapacity);

            this.initialChunkPower = Math.max(MIN_CHUNK_POWER, Integer.SIZE - Integer.numberOfLeadingZeros(initialCapacity - 1));
            curChunk = newArray(1 << initialChunkPower);
        }

        OfPrimitive() {
            this.initialChunkPower = MIN_CHUNK_POWER;
            curChunk = newArray(1 << initialChunkPower);
        }

        @Override
        public abstract Iterator iterator();

        protected abstract T_ARR[] newArrayArray(int size);

        protected abstract T_ARR newArray(int size);

        protected abstract int arrayLength(T_ARR array);

        /**
         * Is the buffer currently empty?
         */
        public boolean isEmpty() {
            return (spineIndex == 0) && (elementIndex == 0);
        }

        /**
         * How many elements are currently in the buffer?
         */
        public long count() {
            return (spineIndex == 0) ? elementIndex : priorElementCount[spineIndex] + elementIndex;
        }

        /**
         * How big should the nth chunk be?
         */
        int chunkSize(int n) {
            int power = (n == 0 || n == 1) ? initialChunkPower : Math.min(initialChunkPower + n - 1, MAX_CHUNK_POWER);
            return 1 << power;
        }

        long capacity() {
            return (spineIndex == 0) ? arrayLength(curChunk) : priorElementCount[spineIndex] + arrayLength(spine[spineIndex]);
        }

        private void inflateSpine() {
            if (spine == null) {
                spine = newArrayArray(MIN_SPINE_SIZE);
                priorElementCount = new long[MIN_SPINE_SIZE];
                spine[0] = curChunk;
            }
        }

        final void ensureCapacity(long targetSize) {
            long capacity = capacity();
            if (targetSize > capacity) {
                inflateSpine();
                for (int i = spineIndex + 1; targetSize > capacity; i++) {
                    if (i >= spine.length) {
                        int newSpineSize = spine.length * 2;
                        spine = Arrays.copyOf(spine, newSpineSize);
                        priorElementCount = Arrays.copyOf(priorElementCount, newSpineSize);
                    }
                    int nextChunkSize = chunkSize(i);
                    spine[i] = newArray(nextChunkSize);
                    priorElementCount[i] = priorElementCount[i - 1] + arrayLength(spine[i - 1]);
                    capacity += nextChunkSize;
                }
            }
        }

        void increaseCapacity() {
            ensureCapacity(capacity() + 1);
        }

        int chunkFor(long index) {
            if (spineIndex == 0) {
                if (index < elementIndex)
                    return 0;
                else
                    throw new IndexOutOfBoundsException(Long.toString(index));
            }

            if (index >= count())
                throw new IndexOutOfBoundsException(Long.toString(index));

            for (int j = 0; j <= spineIndex; j++)
                if (index < priorElementCount[j] + arrayLength(spine[j]))
                    return j;

            throw new IndexOutOfBoundsException(Long.toString(index));
        }

        void copyInto(T_ARR array, int offset) {
            long finalOffset = offset + count();
            if (finalOffset > arrayLength(array) || finalOffset < offset) {
                throw new IndexOutOfBoundsException("does not fit");
            }

            if (spineIndex == 0)
                System.arraycopy(curChunk, 0, array, offset, elementIndex);
            else {
                // full chunks
                for (int i = 0; i < spineIndex; i++) {
                    System.arraycopy(spine[i], 0, array, offset, arrayLength(spine[i]));
                    offset += arrayLength(spine[i]);
                }
                if (elementIndex > 0)
                    System.arraycopy(curChunk, 0, array, offset, elementIndex);
            }
        }

        public T_ARR asPrimitiveArray() {
            long size = count();

            Compat.checkMaxArraySize(size);

            T_ARR result = newArray((int) size);
            copyInto(result, 0);
            return result;
        }

        void preAccept() {
            if (elementIndex == arrayLength(curChunk)) {
                inflateSpine();
                if (spineIndex + 1 >= spine.length || spine[spineIndex + 1] == null)
                    increaseCapacity();
                elementIndex = 0;
                ++spineIndex;
                curChunk = spine[spineIndex];
            }
        }

        /**
         * Remove all data from the buffer
         */
        public void clear() {
            if (spine != null) {
                curChunk = spine[0];
                spine = null;
                priorElementCount = null;
            }
            elementIndex = 0;
            spineIndex = 0;
        }
    }

    static class OfInt extends SpinedBuffer.OfPrimitive implements IntConsumer {
        OfInt() {
        }

        OfInt(int initialCapacity) {
            super(initialCapacity);
        }

        @Override
        protected int[][] newArrayArray(int size) {
            return new int[size][];
        }

        @Override
        public int[] newArray(int size) {
            return new int[size];
        }

        @Override
        protected int arrayLength(int[] array) {
            return array.length;
        }

        @Override
        public void accept(int i) {
            preAccept();
            curChunk[elementIndex++] = i;
        }

        public int get(long index) {
            // Casts to int are safe since the spine array index is the index minus
            // the prior element count from the current spine
            int ch = chunkFor(index);
            if (spineIndex == 0 && ch == 0)
                return curChunk[(int) index];
            else
                return spine[ch][(int) (index - priorElementCount[ch])];
        }

        @Override
        public PrimitiveIterator.OfInt iterator() {
            return new PrimitiveIterator.OfInt() {

                long index = 0;

                @Override
                public int nextInt() {
                    return get(index++);
                }

                @Override
                public boolean hasNext() {
                    return index < count();
                }
            };
        }
    }

    static class OfLong extends SpinedBuffer.OfPrimitive implements LongConsumer {
        OfLong() {
        }

        OfLong(int initialCapacity) {
            super(initialCapacity);
        }

        @Override
        protected long[][] newArrayArray(int size) {
            return new long[size][];
        }

        @Override
        public long[] newArray(int size) {
            return new long[size];
        }

        @Override
        protected int arrayLength(long[] array) {
            return array.length;
        }

        @Override
        public void accept(long i) {
            preAccept();
            curChunk[elementIndex++] = i;
        }

        public long get(long index) {
            int ch = chunkFor(index);
            if (spineIndex == 0 && ch == 0)
                return curChunk[(int) index];
            else
                return spine[ch][(int) (index - priorElementCount[ch])];
        }

        @Override
        public PrimitiveIterator.OfLong iterator() {
            return new PrimitiveIterator.OfLong() {

                long index = 0;

                @Override
                public long nextLong() {
                    return get(index++);
                }

                @Override
                public boolean hasNext() {
                    return index < count();
                }
            };
        }
    }

    static class OfDouble extends SpinedBuffer.OfPrimitive implements DoubleConsumer {
        OfDouble() {
        }

        OfDouble(int initialCapacity) {
            super(initialCapacity);
        }

        @Override
        protected double[][] newArrayArray(int size) {
            return new double[size][];
        }

        @Override
        public double[] newArray(int size) {
            return new double[size];
        }

        @Override
        protected int arrayLength(double[] array) {
            return array.length;
        }

        @Override
        public void accept(double i) {
            preAccept();
            curChunk[elementIndex++] = i;
        }

        public double get(long index) {
            int ch = chunkFor(index);
            if (spineIndex == 0 && ch == 0)
                return curChunk[(int) index];
            else
                return spine[ch][(int) (index - priorElementCount[ch])];
        }

        @Override
        public PrimitiveIterator.OfDouble iterator() {
            return new PrimitiveIterator.OfDouble() {

                long index = 0;

                @Override
                public double nextDouble() {
                    return get(index++);
                }

                @Override
                public boolean hasNext() {
                    return index < count();
                }
            };
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy