package ru.codeinside.gws.xml.normalizer.normalize.utils; import java.io.IOException; import java.io.OutputStream; /** * A simple Unsynced ByteArrayOutputStream * * @author raul */ public class UnsyncByteArrayOutputStream extends OutputStream { private static final int INITIAL_SIZE = 8192; private static ThreadLocal<byte[]> bufCache = new ThreadLocal<byte[]>() { @Override protected synchronized byte[] initialValue() { return new byte[INITIAL_SIZE]; } }; private byte[] buf; private int size = INITIAL_SIZE; private int pos = 0; public UnsyncByteArrayOutputStream() { buf = (byte[]) bufCache.get(); } public void write(byte[] arg0) { if ((Integer.MAX_VALUE - pos) < arg0.length) { throw new OutOfMemoryError(); } int newPos = pos + arg0.length; if (newPos > size) { expandSize(newPos); } System.arraycopy(arg0, 0, buf, pos, arg0.length); pos = newPos; } public void write(byte[] arg0, int arg1, int arg2) { if ((Integer.MAX_VALUE - pos) < arg2) { throw new OutOfMemoryError(); } int newPos = pos + arg2; if (newPos > size) { expandSize(newPos); } System.arraycopy(arg0, arg1, buf, pos, arg2); pos = newPos; } public void write(int arg0) { if ((Integer.MAX_VALUE - pos) == 0) { throw new OutOfMemoryError(); } int newPos = pos + 1; if (newPos > size) { expandSize(newPos); } buf[pos++] = (byte) arg0; } public byte[] toByteArray() { byte result[] = new byte[pos]; System.arraycopy(buf, 0, result, 0, pos); return result; } public void reset() { pos = 0; bufCache.remove(); } @Override public void close() throws IOException { bufCache.remove(); } private void expandSize(int newPos) { int newSize = size; while (newPos > newSize) { newSize = newSize << 1; // Deal with overflow if (newSize < 0) { newSize = Integer.MAX_VALUE; } } byte newBuf[] = new byte[newSize]; System.arraycopy(buf, 0, newBuf, 0, pos); buf = newBuf; size = newSize; } }