21

有没有办法检查BufferedReader对象中是否有要阅读的东西?类似 C++ 的东西cin.peek()。谢谢。

4

8 回答 8

39

您可以使用PushbackReader。使用它,您可以读取一个字符,然后将其取消读取。这基本上允许您将其推回。

PushbackReader pr = new PushbackReader(reader);
char c = (char)pr.read();
// do something to look at c
pr.unread((int)c); //pushes the character back into the buffer
于 2010-03-25T17:05:53.930 回答
19

您可以尝试“布尔就绪()”方法。来自 Java 6 API 文档:“如果缓冲区不为空,或者基础字符流已准备好,则缓冲字符流已准备就绪。”

BufferedReader r = new BufferedReader(reader);
if(r.ready())
{
   r.read();
}
于 2010-03-25T17:03:49.703 回答
14

以下代码将查看 Stream 中的第一个字节。应该作为你的窥视。

BufferedReader bReader = new BufferedReader(inputStream);
bReader.mark(1);
int byte1 = bReader.read();
bReader.reset();
于 2010-03-25T17:03:29.600 回答
4

正常的习惯用法是在循环中检查 if BufferedReader#readLine()doesn't return null。如果到达流的结尾(例如文件结尾、套接字关闭等),则返回null.

例如

BufferedReader reader = new BufferedReader(someReaderSource);
String line = null;
while ((line = reader.readLine()) != null) {
    // ...
}

如果您不想逐行阅读(这是BufferedReader选择 a 的主要原因),请BufferedReader#ready()改用:

BufferedReader reader = new BufferedReader(someReaderSource);
while (reader.ready()) {
    int data = reader.read();
    // ...
}
于 2010-03-25T17:06:18.890 回答
2
BufferedReader br = new BufferedReader(reader);
br.mark(1);
int firstByte = br.read();
br.reset();
于 2010-03-25T17:03:42.283 回答
1

pgmura 的答案(依赖于ready () 方法)很简单并且有效。但请记住,这是因为 Sun 的方法实现;这并不真正同意文档。如果这种行为很关键,我不会依赖它。请参阅此处http://bugs.sun.com/bugdatabase/view_bug.do?bug_id=4090471 我宁愿选择 PushbackReader 选项。

于 2010-03-25T19:16:50.113 回答
1

您可以使用 aPushBackReader读取一个字符,然后“将其推回”。这样你就可以肯定地知道有东西在那里,而不会影响它的整体状态——“偷看”。

于 2010-03-25T17:06:45.110 回答
1

我的解决方案是.. 扩展 BufferedReader 并将队列用作 buf,然后您可以在队列中使用 peek 方法。

public class PeekBufferedReader extends BufferedReader{

    private Queue<String>       buf;
    private int                 bufSize;

    public PeekBufferedReader(Reader reader, int bufSize) throws IOException {
        super(reader);
        this.bufSize = bufSize;
        buf = Queues.newArrayBlockingQueue(bufSize);
    }

    /**
     * readAheadLimit is set to 1048576. Line which has length over readAheadLimit 
     * will cause IOException.
     * @throws IOException 
     **/
    //public String peekLine() throws IOException {
    //  super.mark(1048576);
    //  String peekedLine = super.readLine();
    //  super.reset();
    //  return peekedLine;
    //}

    /**
     * This method can be implemented by mark and reset methods. But performance of 
     * this implementation is better ( about 2times) than using mark and reset  
     **/
    public String peekLine() throws IOException {
        if (buf.isEmpty()) {
            while (buf.size() < bufSize) {
                String readLine = super.readLine();
                if (readLine == null) {
                    break;
                } else {
                    buf.add(readLine);
                }
            }
        } else {
            return buf.peek();
        }
        if (buf.isEmpty()) {
            return null;
        } else {
            return buf.peek();
        }
    }

    public String readLine() throws IOException {
        if (buf.isEmpty()) {
            while (buf.size() < bufSize) {
                String readLine = super.readLine();
                if (readLine == null) {
                    break;
                } else {
                    buf.add(readLine);
                }
            }
        } else {
            return buf.poll();
        }
        if (buf.isEmpty()) {
            return null;
        } else {
            return buf.poll();
        }
    }
    public boolean isEmpty() throws IOException {
        if (buf.isEmpty()) {
            while (buf.size() < bufSize) {
                String readLine = super.readLine();
                if (readLine == null) {
                    break;
                } else {
                    buf.add(readLine);
                }
            }
        } else {
            return false;
        }
        if (buf.isEmpty()) {
            return true;
        } else {
            return false;
        }
    }
}
于 2016-01-05T06:21:22.687 回答