/* * 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 java.io; import java.util.Arrays; /** * Wraps an existing {@link OutputStream} and <em>buffers</em> the output. * Expensive interaction with the underlying input stream is minimized, since * most (smaller) requests can be satisfied by accessing the buffer alone. The * drawback is that some extra space is required to hold the buffer and that * copying takes place when flushing that buffer, but this is usually outweighed * by the performance benefits. * * <p/>A typical application pattern for the class looks like this:<p/> * * <pre> * BufferedOutputStream buf = new BufferedOutputStream(new FileOutputStream("file.java")); * </pre> * * @see BufferedInputStream */ public class BufferedOutputStream extends FilterOutputStream { /** * The buffer containing the bytes to be written to the target stream. */ protected byte[] buf; /** * The total number of bytes inside the byte array {@code buf}. */ protected int count; /** * Constructs a new {@code BufferedOutputStream}, providing {@code out} with a buffer * of 8192 bytes. * * @param out the {@code OutputStream} the buffer writes to. */ public BufferedOutputStream(OutputStream out) { this(out, 8192); } /** * Constructs a new {@code BufferedOutputStream}, providing {@code out} with {@code size} bytes * of buffer. * * @param out the {@code OutputStream} the buffer writes to. * @param size the size of buffer in bytes. * @throws IllegalArgumentException if {@code size <= 0}. */ public BufferedOutputStream(OutputStream out, int size) { super(out); if (size <= 0) { throw new IllegalArgumentException("size <= 0"); } buf = new byte[size]; } /** * Flushes this stream to ensure all pending data is written out to the * target stream. In addition, the target stream is flushed. * * @throws IOException * if an error occurs attempting to flush this stream. */ @Override public synchronized void flush() throws IOException { checkNotClosed(); flushInternal(); out.flush(); } private void checkNotClosed() throws IOException { if (buf == null) { throw new IOException("BufferedOutputStream is closed"); } } /** * Writes {@code count} bytes from the byte array {@code buffer} starting at * {@code offset} to this stream. If there is room in the buffer to hold the * bytes, they are copied in. If not, the buffered bytes plus the bytes in * {@code buffer} are written to the target stream, the target is flushed, * and the buffer is cleared. * * @param buffer * the buffer to be written. * @param offset * the start position in {@code buffer} from where to get bytes. * @param length * the number of bytes from {@code buffer} to write to this * stream. * @throws IndexOutOfBoundsException * if {@code offset < 0} or {@code length < 0}, or if * {@code offset + length} is greater than the size of * {@code buffer}. * @throws IOException * if an error occurs attempting to write to this stream. * @throws NullPointerException * if {@code buffer} is {@code null}. * @throws ArrayIndexOutOfBoundsException * If offset or count is outside of bounds. */ @Override public synchronized void write(byte[] buffer, int offset, int length) throws IOException { checkNotClosed(); if (buffer == null) { throw new NullPointerException("buffer == null"); } byte[] internalBuffer = buf; if (length >= internalBuffer.length) { flushInternal(); out.write(buffer, offset, length); return; } Arrays.checkOffsetAndCount(buffer.length, offset, length); // flush the internal buffer first if we have not enough space left if (length > (internalBuffer.length - count)) { flushInternal(); } System.arraycopy(buffer, offset, internalBuffer, count, length); count += length; } @Override public synchronized void close() throws IOException { if (buf == null) { return; } try { super.close(); } finally { buf = null; } } /** * Writes one byte to this stream. Only the low order byte of the integer * {@code oneByte} is written. If there is room in the buffer, the byte is * copied into the buffer and the count incremented. Otherwise, the buffer * plus {@code oneByte} are written to the target stream, the target is * flushed, and the buffer is reset. * * @param oneByte * the byte to be written. * @throws IOException * if an error occurs attempting to write to this stream. */ @Override public synchronized void write(int oneByte) throws IOException { checkNotClosed(); if (count == buf.length) { out.write(buf, 0, count); count = 0; } buf[count++] = (byte) oneByte; } /** * Flushes only internal buffer. */ private void flushInternal() throws IOException { if (count > 0) { out.write(buf, 0, count); count = 0; } } }