linux系统中InputStream输入流的方法之reset()和mark()命令的注意事项
今天写一个读写程序,运用到inputstream的reset方法是发现竟然失败了,然后查了一下jdk源码,发现bufferinputstream重写了父类filterinputstream的mark和resetf方法,其有支持 mark 和 reset 方法的能力。而fileinputstream则没有重写父类inputstream的这两个方法,其不具有mark和reset方法的能力。
在jdk源码中,写到public synchronized void mark(int readlimit)在该输入流中标记当前位置。 后续调用 reset 方法重新将流定位于最后标记位置,以便后续读取能重新读取相同字节。
readlimit 参数给出当前输入流在标记位置变为非法前允许读取的字节数。
这句话的意思是说:mark就像书签一样,用于标记,以后再调用reset时就可以再回到这个mark过的地方。mark方法有个参数,通过这个整型参数,你告诉系统,希望在读出这么多个字符之前,这个mark保持有效。比如说mark(10),那么在read()10个以内的字符时,reset()操作后可以重新读取已经读出的数据,如果已经读取的数据超过10个,那reset()操作后,就不能正确读取以前的数据了,因为此时mark标记已经失效。
下面是bufferinputstream以及父类filterinputstream 中这两个方法的默认实现//filterinputstream.java
public synchronized void mark(int readlimit) {in.mark(readlimit);
}
public synchronized void reset() throws ioexception {in.reset();
}
<br>public boolean marksupported() {<br> return in.marksupported()<br>}
//bufferedinputstream.java
public synchronized void mark(int readlimit) {marklimit = readlimit;
markpos = pos;
}
public synchronized void reset() throws ioexception {getbufifopen(); // cause exception if closedif (markpos < 0)
throw new ioexception("resetting to invalid mark");pos = markpos;
}<br><br>
public boolean marksupported() {
return true;
} //支持mark操作
而fileinputstream没有重写父类inputstream的默认实现,其默认实现如下:
//inputstream.java
public synchronized void mark(int readlimit) {}//空实现什么都没有实现public synchronized void reset() throws ioexception {throw new ioexception("mark/reset not supported");}//不支持reset操作
public boolean marksupported() {
return false;
} //不支持mark操作
下面是一个简单的程序,用于设置reset和mark.
import java.io.bufferedinputstream;
import java.io.file;
import java.io.fileinputstream;
import java.io.ioexception;
import java.io.inputstream;
public class testinputstream {
public void read(inputstream in) throws ioexception {if(in == null) {
return;
}
int len=0;
in.mark(1);
in.read() //第一次读取
in.reset();//又可以重新读取
in.read()//第二次读取跟第一次读取结果一样。因为只读了一个,没有超过mark设置的整数。所以mark有效}
public static void main(string[] args) throws ioexception {testinputstream test = new testinputstream();string filename = "f:/google.txt";
inputstream in1 = new fileinputstream(new file(filename));if(!in1.marksupported()) {
in1 = new bufferedinputstream(in1);
}
test.read(in1);
byte[] buf = new byte[100];
while(in1.read(buf, 0, buf.length)!=-1) {system.out.println(buf);
}
system.out.println("success!");
}
}
总结:
mark
public void mark(int readlimit)
mark 的常规协定是:如果方法 marksupported 返回 true,那么输入流总是在调用 mark 之后记录所有读取的字节,并时刻准备在调用方法 reset 时(无论何时),再次提供这些相同的字节。但是,如果在调用 reset 之前可以从流中读取多于 readlimit 的字节,则不需要该流记录任何数据。
参数:
readlimit - 在标记位置失效前可以读取字节的最大限制。
4 reset
public void reset() throws ioexception
将此流重新定位到最后一次对此输入流调用 mark 方法时的位置。
reset 的常规协定是:
1、如果方法 marksupported 返回 true,那么:
如果创建流以后未调用方法 mark,或最后调用 mark 以后从该流读取的字节数大于最后调用 mark 时的参数,则可能抛出 ioexception。
如果未抛出这样的 ioexception,则将该流重新设置为这种状态:最近一次调用 mark 以后(如果未调用过 mark,则从文件开头开始)读取的所有字节将重新提供给 read 方法的后续调用者,后跟任何从调用 reset 时起将作为下一输入数据的字节。
2、如果方法 marksupported 返回 false,那么:
对 reset 的调用可能抛出 ioexception。
如果未抛出 ioexception,则将该流重新设置为一种固定状态,该状态取决于输入流的特定类型及其创建方式。提供给 read 方法后续调用者的字节取决于特定类型的输入流。
除了抛出 ioexception 之外,类 inputstream 的方法 reset 不执行任何操作。
抛出:
ioexception - 如果未标记此流或该标记失效。
以上就是在linux系统中inputstream输入流的方法的注意事项,谢谢阅读,希望能帮到大家,请继续关注,我们会努力分享更多优秀的文章。