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

java.io.BufferedReader Maven / Gradle / Ivy

/*
 *  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;

/**
 * BufferedReader is a buffered character input reader. Buffering allows reading
 * from character streams more efficiently. If the default size of the buffer is
 * not practical, another size may be specified. Reading a character from a
 * Reader class usually involves reading a character from its Stream or
 * subsequent Reader. It is advisable to wrap a BufferedReader around those
 * Readers whose read operations may have high latency. For example, the
 * following code
 *
 * 
 * BufferedReader inReader = new BufferedReader(new FileReader("file.java"));
 * 
* * will buffer input for the file file.java. * * @see BufferedWriter * @since 1.1 */ public class BufferedReader extends Reader { private Reader in2; private char[] buf; private int marklimit= -1; private int count; private int markpos= -1; private int pos; /** * Constructs a new BufferedReader on the Reader in. The * default buffer size (8K) is allocated and all reads can now be filtered * through this BufferedReader. * * @param in * the Reader to buffer reads on. */ public BufferedReader(Reader in) { super(in); this.in2= in; buf= new char[8192]; } /** * Constructs a new BufferedReader on the Reader in. The * buffer size is specified by the parameter size and all * reads can now be filtered through this BufferedReader. * * @param in * the Reader to buffer reads on. * @param size * the size of buffer to allocate. * @throws IllegalArgumentException * if the size is <= 0 */ public BufferedReader(Reader in, int size) { super(in); if (size > 0) { this.in2= in; buf= new char[size]; } else { throw new IllegalArgumentException(Msg.getString("K0058")); //$NON-NLS-1$ } } /** * Close the Reader. This implementation closes the Reader being filtered * and releases the buffer used by this reader. If this BufferedReader has * already been closed, nothing is done. * * @throws IOException * If an error occurs attempting to close this BufferedReader. */ @Override public void close() throws IOException { synchronized (lock) { if (isOpen()) { in2.close(); buf= null; } } } private int fillbuf() throws IOException { if (markpos == -1 || (pos - markpos >= marklimit)) { /* Mark position not set or exceeded readlimit */ int result= in2.read(buf, 0, buf.length); if (result > 0) { markpos= -1; pos= 0; count= result == -1 ? 0 : result; } return result; } if (markpos == 0 && marklimit > buf.length) { /* Increase buffer size to accommodate the readlimit */ int newLength= buf.length * 2; if (newLength > marklimit) { newLength= marklimit; } char[] newbuf= new char[newLength]; System.arraycopy(buf, 0, newbuf, 0, buf.length); buf= newbuf; } else if (markpos > 0) { System.arraycopy(buf, markpos, buf, 0, buf.length - markpos); } /* Set the new position and mark position */ pos-= markpos; count= markpos= 0; int charsread= in2.read(buf, pos, buf.length - pos); count= charsread == -1 ? pos : pos + charsread; return charsread; } /** * Answer a boolean indicating whether or not this BufferedReader is open. * * @return true if this reader is open, false * otherwise */ private boolean isOpen() { return buf != null; } /** * Set a Mark position in this BufferedReader. The parameter * readLimit indicates how many characters can be read before * a mark is invalidated. Sending reset() will reposition the reader back to * the marked position provided readLimit has not been * surpassed. * * @param readlimit * an int representing how many characters must be read * before invalidating the mark. * * @throws IOException * If an error occurs attempting mark this BufferedReader. * @throws IllegalArgumentException * If readlimit is < 0 */ @Override public void mark(int readlimit) throws IOException { if (readlimit >= 0) { synchronized (lock) { if (isOpen()) { marklimit= readlimit; markpos= pos; } else { throw new IOException(Msg.getString("K005b")); //$NON-NLS-1$ } } } else { throw new IllegalArgumentException(); } } /** * Answers a boolean indicating whether or not this Reader supports mark() * and reset(). This implementation answers true. * * @return true if mark() and reset() are supported, * false otherwise */ @Override public boolean markSupported() { return true; } /** * Reads a single character from this reader and returns the result as an * int. The 2 higher-order characters are set to 0. If the end of reader was * encountered then return -1. This implementation either returns a * character from the buffer or if there are no characters available, fill * the buffer then return a character or -1. * * @return the character read or -1 if end of reader. * * @throws IOException * If the BufferedReader is already closed or some other IO * error occurs. */ @Override public int read() throws IOException { synchronized (lock) { if (isOpen()) { /* Are there buffered characters available? */ if (pos < count || fillbuf() != -1) { return buf[pos++]; } return -1; } throw new IOException(Msg.getString("K005b")); //$NON-NLS-1$ } } /** * Reads at most length characters from this BufferedReader * and stores them at offset in the character array * buffer. Returns the number of characters actually read or * -1 if the end of reader was encountered. If all the buffered characters * have been used, a mark has not been set, and the requested number of * characters is larger than this Readers buffer size, this implementation * bypasses the buffer and simply places the results directly into * buffer. * * @param buffer * character array to store the read characters * @param offset * offset in buf to store the read characters * @param length * maximum number of characters to read * @return number of characters read or -1 if end of reader. * * @throws IOException * If the BufferedReader is already closed or some other IO * error occurs. */ @Override public int read(char[] buffer, int offset, int length) throws IOException { synchronized (lock) { if (!isOpen()) { throw new IOException(Msg.getString("K005b")); //$NON-NLS-1$ } if (offset < 0 || offset > buffer.length - length || length < 0) { throw new IndexOutOfBoundsException(); } if (length == 0) { return 0; } int required; if (pos < count) { /* There are bytes available in the buffer. */ int copylength= count - pos >= length ? length : count - pos; System.arraycopy(buf, pos, buffer, offset, copylength); pos+= copylength; if (copylength == length || !in2.ready()) { return copylength; } offset+= copylength; required= length - copylength; } else { required= length; } while (true) { int read; /* * If we're not marked and the required size is greater than the * buffer, simply read the bytes directly bypassing the buffer. */ if (markpos == -1 && required >= buf.length) { read= in2.read(buffer, offset, required); if (read == -1) { return required == length ? -1 : length - required; } } else { if (fillbuf() == -1) { return required == length ? -1 : length - required; } read= count - pos >= required ? required : count - pos; System.arraycopy(buf, pos, buffer, offset, read); pos+= read; } required-= read; if (required == 0) { return length; } if (!in2.ready()) { return length - required; } offset+= read; } } } /** * Answers a String representing the next line of text * available in this BufferedReader. A line is represented by 0 or more * characters followed by '\n', '\r', * '\r\n' or end of stream. The String does * not include the newline sequence. * * @return the contents of the line or null if no characters were read * before end of stream. * * @throws IOException * If the BufferedReader is already closed or some other IO * error occurs. */ public String readLine() throws IOException { synchronized (lock) { if (isOpen()) { /* Are there buffered characters available? */ if ((pos >= count) && (fillbuf() == -1)) { return null; } for (int charPos= pos; charPos < count; charPos++) { char ch= buf[charPos]; if (ch > '\r') continue; if (ch == '\n') { String res= new String(buf, pos, charPos - pos); pos= charPos + 1; return res; } else if (ch == '\r') { String res= new String(buf, pos, charPos - pos); pos= charPos + 1; if (((pos < count) || (fillbuf() != -1)) && (buf[pos] == '\n')) { pos++; } return res; } } char eol= '\0'; StringBuilder result= new StringBuilder(80); /* Typical Line Length */ result.append(buf, pos, count - pos); pos= count; while (true) { /* Are there buffered characters available? */ if (pos >= count) { if (eol == '\n') { return result.toString(); } // attempt to fill buffer if (fillbuf() == -1) { // characters or null. return result.length() > 0 || eol != '\0' ? result.toString() : null; } } for (int charPos= pos; charPos < count; charPos++) { if (eol == '\0') { if ((buf[charPos] == '\n' || buf[charPos] == '\r')) { eol= buf[charPos]; } } else if (eol == '\r' && (buf[charPos] == '\n')) { if (charPos > pos) { result.append(buf, pos, charPos - pos - 1); } pos= charPos + 1; return result.toString(); } else if (eol != '\0') { if (charPos > pos) { result.append(buf, pos, charPos - pos - 1); } pos= charPos; return result.toString(); } } if (eol == '\0') { result.append(buf, pos, count - pos); } else { result.append(buf, pos, count - pos - 1); } pos= count; } } throw new IOException(Msg.getString("K005b")); //$NON-NLS-1$ } } /** * Answers a boolean indicating whether or not this Reader is * ready to be read without blocking. If the result is true, * the next read() will not block. If the result is * false this Reader may or may not block when * read() is sent. * * @return true if the receiver will not block when * read() is called, false if unknown * or blocking will occur. * * @throws IOException * If the BufferedReader is already closed or some other IO * error occurs. */ @Override public boolean ready() throws IOException { synchronized (lock) { if (isOpen()) { return ((count - pos) > 0) || in2.ready(); } throw new IOException(Msg.getString("K005b")); //$NON-NLS-1$ } } /** * Reset this BufferedReader's position to the last mark() * location. Invocations of read()/skip() will occur from * this new location. If this Reader was not marked, throw IOException. * * @throws IOException * If a problem occurred, the receiver does not support * mark()/reset(), or no mark has been set. */ @Override public void reset() throws IOException { synchronized (lock) { if (isOpen()) { if (markpos != -1) { pos= markpos; } else { throw new IOException(Msg.getString("K005c")); //$NON-NLS-1$ } } else { throw new IOException(Msg.getString("K005b")); //$NON-NLS-1$ } } } /** * Skips amount number of characters in this Reader. * Subsequent read()'s will not return these characters * unless reset() is used. Skipping characters may invalidate * a mark if marklimit is surpassed. * * @param amount * the maximum number of characters to skip. * @return the number of characters actually skipped. * * @throws IOException * If the BufferedReader is already closed or some other IO * error occurs. * @throws IllegalArgumentException * If amount is negative */ @Override public long skip(long amount) throws IOException { if (amount >= 0) { synchronized (lock) { if (isOpen()) { if (amount < 1) { return 0; } if (count - pos >= amount) { pos+= amount; return amount; } long read= count - pos; pos= count; while (read < amount) { if (fillbuf() == -1) { return read; } if (count - pos >= amount - read) { pos+= amount - read; return amount; } // Couldn't get all the characters, skip what we read read+= (count - pos); pos= count; } return amount; } throw new IOException(Msg.getString("K005b")); //$NON-NLS-1$ } } throw new IllegalArgumentException(); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy