16

mark()和方法如何reset()准确地工作(在下面的代码中),一步一步?我尝试编写自己的示例,但开始抛出错误的标记异常或类似的异常,并且我无法理解在此代码中放置标记和重置方法的意义何在,因为我看不出有无区别。

import java.io.*;

class BufferedInputStreamDemo {
    public static void main(String args[]) {
        String s = "© is a copyright symbol, "
                + "however &copy isn't.\n";
        byte buf[] = s.getBytes();

        ByteArrayInputStream in = new ByteArrayInputStream(buf);
        int c;
        boolean marked = false;

        //try_with_resources
        try (BufferedInputStream f = new BufferedInputStream(in)) {
            while ((c = f.read()) != -1) {
                switch (c) {
                    case '&':
                        if (!marked) {
                            f.mark(32);
                            marked = true;
                        } else {
                            marked = false;
                        }
                        break;
                    case ';':
                        if (marked) {
                            marked = false;
                            System.out.print("(c)");
                        } else
                            System.out.print((char) c);
                        break;
                    case ' ':
                        if (marked) {
                            marked = false;
                            f.reset();
                            System.out.print("&");
                        } else
                            System.out.print((char) c);
                        break;
                    default:
                        if (!marked)
                            System.out.print((char) c);
                        break;
                }
            }
        } catch (IOException e) {
            System.out.println("I/O Error: " + e);
        }
    }
}
4

2 回答 2

8

到达时f.mark(32);,读取光标已经在 之后&,并且设置了一个标记reset以知道从哪里跳回来。因此,当您检测到;缺少 a 来关闭元素时,您将使用该方法手动打印&并向右移动读取光标(在&放置标记的位置之后,使用mark(32)调用)reset。在下一次读取时,因为您的marked变量未设置,它将打印字符。

mark(32)表示如果您的阅读光标将前进超过 32 个字符,则自动删除标记。这可能是您的其他代码中的问题,即触发错误,因为标记已经失效。

于 2012-11-02T14:15:50.727 回答
6

请参阅 API 文档:

mark(int)

标记此输入流中的当前位置。对 reset 方法的后续调用将此流重新定位到最后标记的位置,以便后续读取重新读取相同的字节。

readlimit 参数告诉此输入流允许在标记位置无效之前读取那么多字节。

这个方法简单地执行 in.mark(readlimit)。

reset()

将此流重新定位到最后一次在此输入流上调用标记方法时的位置。

这个方法只是简单地执行 in.reset()。

流标记旨在用于需要提前阅读以查看流中内容的情况。通常这最容易通过调用一些通用解析器来完成。如果流是由解析处理的类型,它就会愉快地前进。如果流不是那种类型,解析器应该在失败时抛出一个异常。如果这发生在 readlimit 字节内,它允许外部代码重置流并尝试另一个解析器。

于 2012-11-02T14:07:16.057 回答