CircularByteBuffer.java
/*
* 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.commons.io.input.buffer;
import java.util.Objects;
import org.apache.commons.io.IOUtils;
/**
* A buffer, which doesn't need reallocation of byte arrays, because it
* reuses a single byte array. This works particularly well, if reading
* from the buffer takes place at the same time than writing to. Such is the
* case, for example, when using the buffer within a filtering input stream,
* like the {@link CircularBufferInputStream}.
*
* @since 2.7
*/
public class CircularByteBuffer {
private final byte[] buffer;
private int startOffset;
private int endOffset;
private int currentNumberOfBytes;
/**
* Constructs a new instance with a reasonable default buffer size ({@link IOUtils#DEFAULT_BUFFER_SIZE}).
*/
public CircularByteBuffer() {
this(IOUtils.DEFAULT_BUFFER_SIZE);
}
/**
* Constructs a new instance with the given buffer size.
*
* @param size the size of buffer to create
*/
public CircularByteBuffer(final int size) {
buffer = IOUtils.byteArray(size);
startOffset = 0;
endOffset = 0;
currentNumberOfBytes = 0;
}
/**
* Adds a new byte to the buffer, which will eventually be returned by following
* invocations of {@link #read()}.
*
* @param value The byte, which is being added to the buffer.
* @throws IllegalStateException The buffer is full. Use {@link #hasSpace()},
* or {@link #getSpace()}, to prevent this exception.
*/
public void add(final byte value) {
if (currentNumberOfBytes >= buffer.length) {
throw new IllegalStateException("No space available");
}
buffer[endOffset] = value;
++currentNumberOfBytes;
if (++endOffset == buffer.length) {
endOffset = 0;
}
}
/**
* Adds the given bytes to the buffer. This is the same as invoking {@link #add(byte)}
* for the bytes at offsets {@code offset+0}, {@code offset+1}, ...,
* {@code offset+length-1} of byte array {@code targetBuffer}.
*
* @param targetBuffer the buffer to copy
* @param offset start offset
* @param length length to copy
* @throws IllegalStateException The buffer doesn't have sufficient space. Use
* {@link #getSpace()} to prevent this exception.
* @throws IllegalArgumentException Either of {@code offset}, or {@code length} is negative.
* @throws NullPointerException The byte array {@code pBuffer} is null.
*/
public void add(final byte[] targetBuffer, final int offset, final int length) {
Objects.requireNonNull(targetBuffer, "Buffer");
if (offset < 0 || offset >= targetBuffer.length) {
throw new IllegalArgumentException("Illegal offset: " + offset);
}
if (length < 0) {
throw new IllegalArgumentException("Illegal length: " + length);
}
if (currentNumberOfBytes + length > buffer.length) {
throw new IllegalStateException("No space available");
}
for (int i = 0; i < length; i++) {
buffer[endOffset] = targetBuffer[offset + i];
if (++endOffset == buffer.length) {
endOffset = 0;
}
}
currentNumberOfBytes += length;
}
/**
* Removes all bytes from the buffer.
*/
public void clear() {
startOffset = 0;
endOffset = 0;
currentNumberOfBytes = 0;
}
/**
* Gets the number of bytes, that are currently present in the buffer.
*
* @return the number of bytes
*/
public int getCurrentNumberOfBytes() {
return currentNumberOfBytes;
}
/**
* Gets the number of bytes, that can currently be added to the buffer.
*
* @return the number of bytes that can be added
*/
public int getSpace() {
return buffer.length - currentNumberOfBytes;
}
/**
* Tests whether the buffer is currently holding at least a single byte.
*
* @return true whether the buffer is currently holding at least a single byte.
*/
public boolean hasBytes() {
return currentNumberOfBytes > 0;
}
/**
* Tests whether there is currently room for a single byte in the buffer.
* Same as {@link #hasSpace(int) hasSpace(1)}.
*
* @return true whether there is currently room for a single byte in the buffer.
* @see #hasSpace(int)
* @see #getSpace()
*/
public boolean hasSpace() {
return currentNumberOfBytes < buffer.length;
}
/**
* Tests whether there is currently room for the given number of bytes in the buffer.
*
* @param count the byte count
* @return true whether there is currently room for the given number of bytes in the buffer.
* @see #hasSpace()
* @see #getSpace()
*/
public boolean hasSpace(final int count) {
return currentNumberOfBytes + count <= buffer.length;
}
/**
* Returns, whether the next bytes in the buffer are exactly those, given by
* {@code sourceBuffer}, {@code offset}, and {@code length}. No bytes are being
* removed from the buffer. If the result is true, then the following invocations
* of {@link #read()} are guaranteed to return exactly those bytes.
*
* @param sourceBuffer the buffer to compare against
* @param offset start offset
* @param length length to compare
* @return True, if the next invocations of {@link #read()} will return the
* bytes at offsets {@code pOffset}+0, {@code pOffset}+1, ...,
* {@code pOffset}+{@code length}-1 of byte array {@code pBuffer}.
* @throws IllegalArgumentException Either of {@code pOffset}, or {@code length} is negative.
* @throws NullPointerException The byte array {@code pBuffer} is null.
*/
public boolean peek(final byte[] sourceBuffer, final int offset, final int length) {
Objects.requireNonNull(sourceBuffer, "Buffer");
if (offset < 0 || offset >= sourceBuffer.length) {
throw new IllegalArgumentException("Illegal offset: " + offset);
}
if (length < 0 || length > buffer.length) {
throw new IllegalArgumentException("Illegal length: " + length);
}
if (length < currentNumberOfBytes) {
return false;
}
int localOffset = startOffset;
for (int i = 0; i < length; i++) {
if (buffer[localOffset] != sourceBuffer[i + offset]) {
return false;
}
if (++localOffset == buffer.length) {
localOffset = 0;
}
}
return true;
}
/**
* Returns the next byte from the buffer, removing it at the same time, so
* that following invocations won't return it again.
*
* @return The byte, which is being returned.
* @throws IllegalStateException The buffer is empty. Use {@link #hasBytes()},
* or {@link #getCurrentNumberOfBytes()}, to prevent this exception.
*/
public byte read() {
if (currentNumberOfBytes <= 0) {
throw new IllegalStateException("No bytes available.");
}
final byte b = buffer[startOffset];
--currentNumberOfBytes;
if (++startOffset == buffer.length) {
startOffset = 0;
}
return b;
}
/**
* Returns the given number of bytes from the buffer by storing them in
* the given byte array at the given offset.
*
* @param targetBuffer The byte array, where to add bytes.
* @param targetOffset The offset, where to store bytes in the byte array.
* @param length The number of bytes to return.
* @throws NullPointerException The byte array {@code pBuffer} is null.
* @throws IllegalArgumentException Either of {@code pOffset}, or {@code length} is negative,
* or the length of the byte array {@code targetBuffer} is too small.
* @throws IllegalStateException The buffer doesn't hold the given number
* of bytes. Use {@link #getCurrentNumberOfBytes()} to prevent this
* exception.
*/
public void read(final byte[] targetBuffer, final int targetOffset, final int length) {
Objects.requireNonNull(targetBuffer, "targetBuffer");
if (targetOffset < 0 || targetOffset >= targetBuffer.length) {
throw new IllegalArgumentException("Illegal offset: " + targetOffset);
}
if (length < 0 || length > buffer.length) {
throw new IllegalArgumentException("Illegal length: " + length);
}
if (targetOffset + length > targetBuffer.length) {
throw new IllegalArgumentException("The supplied byte array contains only "
+ targetBuffer.length + " bytes, but offset, and length would require "
+ (targetOffset + length - 1));
}
if (currentNumberOfBytes < length) {
throw new IllegalStateException("Currently, there are only " + currentNumberOfBytes
+ "in the buffer, not " + length);
}
int offset = targetOffset;
for (int i = 0; i < length; i++) {
targetBuffer[offset++] = buffer[startOffset];
--currentNumberOfBytes;
if (++startOffset == buffer.length) {
startOffset = 0;
}
}
}
}