Open Source Repository

Home /jodd/jodd-3.3.2 | Repository Home



jodd/util/buffer/FastLongBuffer.java
// Copyright (c) 2003-2012, Jodd Team (jodd.org). All Rights Reserved.

package jodd.util.buffer;

/**
 * Fast, fast <code>long</code> buffer.
 */
public class FastLongBuffer {

  private long[][] buffers = new long[16][];
  private int buffersCount;
  private int currentBufferIndex = -1;
  private long[] currentBuffer;
  private int offset;
  private int count;

  /**
   * Creates a new <code>long</code> buffer. The buffer capacity is
   * initially 1024 bytes, though its size increases if necessary.
   */
  public FastLongBuffer() {
    this(1024);
  }

  /**
   * Creates a new <code>long</code> buffer, with a buffer capacity of
   * the specified size, in bytes.
   *
   @param size the initial size.
   @throws IllegalArgumentException if size is negative.
   */
  public FastLongBuffer(int size) {
    if (size < 0) {
      throw new IllegalArgumentException("Invalid size: " + size);
    }
    needNewBuffer(size);
  }

  private void needNewBuffer(int newCount) {
    if (currentBufferIndex < buffersCount - 1) {  // recycling old buffer
      offset = 0;
      currentBufferIndex++;
      currentBuffer = buffers[currentBufferIndex];
    else {                    // creating new buffer
      int newBufferSize;
      if (currentBuffer == null) {
        newBufferSize = newCount;
      else {
        newBufferSize = Math.max(
            currentBuffer.length << 1,
            newCount - count);    // this will give no free additional space

      }

      currentBufferIndex++;
      currentBuffer = new long[newBufferSize];
      offset = 0;

      // add buffer
      if (currentBufferIndex >= buffers.length) {
        int newLen = buffers.length << 1;
        long[][] newBuffers = new long[newLen][];
                System.arraycopy(buffers, 0, newBuffers, 0, buffers.length);
                buffers = newBuffers;
      }
      buffers[currentBufferIndex= currentBuffer;
      buffersCount++;
    }
  }

  /**
   * Appends <code>long</code> array to buffer.
   */
  public FastLongBuffer append(long[] array, int off, int len) {
    int end = off + len;
    if ((off < 0)
        || (off > array.length)
        || (len < 0)
        || (end > array.length)
        || (end < 0)) {
      throw new IndexOutOfBoundsException();
    }
    if (len == 0) {
      return this;
    }
    int newCount = count + len;
    int remaining = len;
    while (remaining > 0) {
      int part = Math.min(remaining, currentBuffer.length - offset);
      System.arraycopy(array, end - remaining, currentBuffer, offset, part);
      remaining -= part;
      offset += part;
      count += part;
      if (remaining > 0) {
        needNewBuffer(newCount);
      }
    }
    return this;
  }

  /**
   * Appends <code>long</code> array to buffer.
   */
  public FastLongBuffer append(long[] array) {
    return append(array, 0, array.length);
  }

  /**
   * Appends single <code>long</code> to buffer.
   */
  public FastLongBuffer append(long element) {
    if (offset == currentBuffer.length) {
      needNewBuffer(count + 1);
    }

    currentBuffer[offset= element;
    offset++;
    count++;

    return this;
  }

  /**
   * Appends another fast buffer to this one.
   */
  public FastLongBuffer append(FastLongBuffer buff) {
    for (int i = 0; i < buff.currentBufferIndex; i++) {
      append(buff.buffers[i]);
    }
    append(buff.currentBuffer, 0, buff.offset);
    return this;
  }

  /**
   * Returns buffer size.
   */
  public int size() {
    return count;
  }

  /**
   * Tests if this buffer has no elements.
   */
  public boolean isEmpty() {
    return count == 0;
  }

  /**
   * Returns current index of inner <code>long</code> array chunk.
   * Represents the index of last used inner array chunk.
   */
  public int index() {
    return currentBufferIndex;
  }

  /**
   * Returns the offset of last used element in current inner array chunk.
   */
  public int offset() {
    return offset;
  }

  /**
   * Returns <code>long</code> inner array chunk at given index.
   * May be used for iterating inner chunks in fast manner.
   */
  public long[] array(int index) {
    return buffers[index];
  }

  /**
   * Resets the buffer content.
   */
  public void clear() {
    count = 0;
    offset = 0;
    currentBufferIndex = 0;
    currentBuffer = buffers[currentBufferIndex];
    buffersCount = 1;
  }

  /**
   * Creates <code>long</code> array from buffered content.
   */
  public long[] toArray() {
    int remaining = count;
    int pos = 0;
    long[] array = new long[count];
    for (long[] buf : buffers) {
      int c = Math.min(buf.length, remaining);
      System.arraycopy(buf, 0, array, pos, c);
      pos += c;
      remaining -= c;
      if (remaining == 0) {
        break;
      }
    }
    return array;
  }

    /**
     * Creates <code>long</code> subarray from buffered content.
     */
  public long[] toArray(int start, int len) {
    int remaining = len;
    int pos = 0;
    long[] array = new long[len];

    if (len == 0) {
      return array;
    }

    int i = 0;
    while (start >= buffers[i].length) {
      start -= buffers[i].length;
      i++;
    }

    while (i < buffersCount) {
      long[] buf = buffers[i];
      int c = Math.min(buf.length - start, remaining);
      System.arraycopy(buf, start, array, pos, c);
      pos += c;
      remaining -= c;
      if (remaining == 0) {
        break;
      }
      start = 0;
      i++;
    }
    return array;
  }

  /**
   * Returns <code>long</code> element at given index.
   */
  public long get(int index) {
    if (index >= count) {
      throw new IndexOutOfBoundsException();
    }
    int ndx = 0;
        while (true) {
      long[] b = buffers[ndx];
      if (index < b.length) {
        return b[index];
      }
      ndx++;
      index -= b.length;
    }
  }

}