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

org.spf4j.io.ByteArrayBuilder Maven / Gradle / Ivy

Go to download

A continuously growing collection of utilities to measure performance, get better diagnostics, improve performance, or do things more reliably, faster that other open source libraries...

The newest version!
/*
 * Copyright (c) 2001-2017, Zoltan Farkas All Rights Reserved.
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this program; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
 *
 * Additionally licensed with:
 *
 * 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 org.spf4j.io;

import edu.umd.cs.findbugs.annotations.CleanupObligation;
import edu.umd.cs.findbugs.annotations.DischargesObligation;
import edu.umd.cs.findbugs.annotations.SuppressFBWarnings;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.nio.charset.Charset;
import java.nio.charset.StandardCharsets;
import java.util.Arrays;
import javax.annotation.Nullable;
import org.spf4j.recyclable.SizedRecyclingSupplier;
import org.spf4j.recyclable.impl.ArraySuppliers;

/**
 * Utility class to avoid replicating byte arrays for no good reason.
 *
 * @author zoly
 */
@SuppressFBWarnings("EI_EXPOSE_REP")
@CleanupObligation
public final class ByteArrayBuilder extends OutputStream {

  /**
   * The maximum size of array to allocate. Some VMs reserve some header words in an array. Attempts to allocate larger
   * arrays may result in OutOfMemoryError: Requested array size exceeds VM limit
   */
  private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8;

  /**
   * The buffer where data is stored.
   */
  private byte[] buf;

  /**
   * The number of valid bytes in the buffer.
   */
  private int count;

  @Nullable
  private final SizedRecyclingSupplier arraySupplier;

  public ByteArrayBuilder() {
    this(256, ArraySuppliers.Bytes.TL_SUPPLIER);
  }

  public ByteArrayBuilder(final int size) {
    this(size, ArraySuppliers.Bytes.TL_SUPPLIER);
  }

  public synchronized byte[] getBuffer() {
    return buf;
  }

  /**
   * Creates a new byte array output stream, with a buffer capacity of the specified size, in bytes.
   *
   * @param size the initial size.
   * @exception IllegalArgumentException if size is negative.
   */
  public ByteArrayBuilder(final int size, @Nullable final SizedRecyclingSupplier arraySupplier) {
    if (size < 0) {
      throw new IllegalArgumentException("Negative initial size: "
              + size);
    }
    this.arraySupplier = arraySupplier;
    if (arraySupplier == null) {
      if (size == 0) {
        buf = org.spf4j.base.Arrays.EMPTY_BYTE_ARRAY;
      } else {
        buf = new byte[size];
      }
    } else {
      buf = arraySupplier.get(size);
    }
  }

  /**
   * Increases the capacity if necessary to ensure that it can hold at least the number of elements specified by the
   * minimum capacity argument.
   *
   * @param minCapacity the desired minimum capacity
   * @throws OutOfMemoryError if {@code minCapacity < 0}. This is interpreted as a request for the unsatisfiably large
   * capacity {@code (long) Integer.MAX_VALUE + (minCapacity - Integer.MAX_VALUE)}.
   */
  private void ensureCapacity(final int minCapacity) {
    // overflow-conscious code
    if (minCapacity - buf.length > 0) {
      grow(minCapacity);
    }
  }

  /**
   * Increases the capacity to ensure that it can hold at least the number of elements specified by the minimum capacity
   * argument.
   *
   * @param minCapacity the desired minimum capacity
   */
  private void grow(final int minCapacity) {
    // overflow-conscious code
    int oldCapacity = buf.length;
    int newCapacity = oldCapacity << 1;
    if (newCapacity - minCapacity < 0) {
      newCapacity = minCapacity;
    }
    if (newCapacity - MAX_ARRAY_SIZE > 0) {
      newCapacity = hugeCapacity(minCapacity);
    }
    if (arraySupplier == null) {
      buf = Arrays.copyOf(buf, newCapacity);
    } else {
      byte[] old = buf;
      buf = arraySupplier.get(newCapacity);
      System.arraycopy(old, 0, buf, 0, old.length);
      arraySupplier.recycle(old);
    }
  }

  private static int hugeCapacity(final int minCapacity) {
    if (minCapacity < 0) { // overflow
      throw new OutOfMemoryError();
    }
    return (minCapacity > MAX_ARRAY_SIZE) ? Integer.MAX_VALUE : MAX_ARRAY_SIZE;
  }

  @Override
  public void write(final byte[] b) {
    write(b, 0, b.length);
  }

  /**
   * Writes the specified byte to this byte array output stream.
   *
   * @param b the byte to be written.
   */
  @Override
  public synchronized void write(final int b) {
    int cp1 = count + 1;
    ensureCapacity(cp1);
    buf[count] = (byte) b;
    count = cp1;
  }

  /**
   * Writes len bytes from the specified byte array starting at offset off to this byte array
   * output stream.
   *
   * @param b the data.
   * @param off the start offset in the data.
   * @param len the number of bytes to write.
   */
  @Override
  public synchronized void write(final byte[] b, final int off, final int len) {
    if ((off < 0) || (off > b.length) || (len < 0) || ((off + len) - b.length > 0)) {
      throw new IndexOutOfBoundsException();
    }
    int cpl = count + len;
    ensureCapacity(cpl);
    System.arraycopy(b, off, buf, count, len);
    count = cpl;
  }

  /**
   * Writes the complete contents of this byte array output stream to the specified output stream argument, as if by
   * calling the output stream's write method using out.write(buf, 0, count).
   *
   * @param out the output stream to which to write the data.
   * @exception IOException if an I/O error occurs.
   */
  public synchronized void writeTo(final OutputStream out) throws IOException {
    out.write(buf, 0, count);
  }

  public synchronized void readFrom(final InputStream in) throws IOException {
    do {
      ensureCapacity(count + 8192);
      int nr = in.read(buf, count, 8192);
      if (nr < 0) {
        break;
      }
      count += nr;
    } while (true);
  }

  /**
   * Resets the count field of this byte array output stream to zero, so that all currently accumulated
   * output in the output stream is discarded. The output stream can be used again, reusing the already allocated buffer
   * space.
   *
   * @see java.io.ByteArrayInputStream#count
   */
  public synchronized void reset() {
    count = 0;
  }

  public synchronized void resetCountTo(final int pos) {
    count = pos;
  }

  /**
   * Creates a newly allocated byte array. Its size is the current size of this output stream and the valid contents of
   * the buffer have been copied into it.
   *
   * @return the current contents of this output stream, as a byte array.
   * @see java.io.ByteArrayOutputStream#size()
   */
  public synchronized byte[] toByteArray() {
    return Arrays.copyOf(buf, count);
  }

  /**
   * Returns the current size of the buffer.
   *
   * @return the value of the count field, which is the number of valid bytes in this output stream.
   * @see java.io.ByteArrayOutputStream#count
   */
  public synchronized int size() {
    return count;
  }

  /**
   * Converts the buffer's contents into a string decoding bytes using the platform's default character set. The length
   * of the new String
   * is a function of the character set, and hence may not be equal to the size of the buffer.
   *
   * 

* This method always replaces malformed-input and unmappable-character sequences with the default replacement string * for the platform's default character set. The {@linkplain java.nio.charset.CharsetDecoder} class should be used * when more control over the decoding process is required. * * @return String decoded from the buffer's contents. */ @Override public synchronized String toString() { return toString(StandardCharsets.UTF_8); } public synchronized String toString(final Charset charset) { if (buf == null) { return "Closed ByteArrayBuilder"; } else { return new String(buf, 0, count, charset); } } /** * Closing a ByteArrayOutputStream will likely recycle the underlying buffer. use of the builder after close * is not advised */ @DischargesObligation @Override public synchronized void close() { if (arraySupplier != null && buf != null) { arraySupplier.recycle(buf); buf = null; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy