java.io (2) 输入流的复用

877 查看

输入流的复用在实际的开发中是很常见的场景。

在实际应用中,很多需要提供输入数据的API都是用IputStream类作为其参数类型,比如XML文档的解析API就是一个典型的例子。同时很多数据的提供者允许使用者通过InputStream类的对象方式来读取数据。

根据java对io的抽象(Stream),流的复用是矛盾的。按照流本身所代表的抽象含义,数据一旦流过去了,就无法被再次利用了。

对于现实应用中存在的对输入流的需求,基本上来说的两种方式可以解决;第一种是利用输入流提供的标记和重置的控制能力,第二种则是把输入流转换成数据来使用。

方法一, 缓冲流类标记与重置

好在java为我们提供现成的解决方案。
BufferedInputStream, BufferedOutputStream
利用这两个类中提供的

public synchronized void mark(int readlimit) 读取位置标记,需要一个允许读取的字节数。
public synchronized void reset() throws IOException 将读取位置重置到标记位置。
public boolean markSupported() 检查当前流类是否支持标记功能

这三个方法能有效地解决流复用的问题。

java   // 假如我们有三个方法,需要串行的使用我们的 io 输入流。
   // public static void process1(InputStream in) throws IOException;
   // public static void process2(InputStream in) throws IOException;
   // public static void process3(InputStream in) throws IOException;

   if (!input.markSupported()) {
       this.input = new BufferedInputStream(input);
   }
   else
   {
       this.input = input;
   }

   // 为了能够复用整个流
   this.input.mark(Integer.MAX_VALUE); 

   process1(this.input);
   // 重置读取位置,以供下一个使用者使用,以下雷同
   this.input.reset();

   process2(this.input);
   this.input.reset();

   process3(this.input);
   this.input.reset();

方法二, 将流转化成为数据流类

这一种方案就是将流中的数据全部读取到一个字节数组中。在不同的数据接收者之间这些数据的传递是通过字节数组完成的。
将 字节流转化成字节数组 可以使用我们 ByteArrayInputStreamByteArrayOutputStream。这样的话,我们传递的是数据,而不再是原始的io。


ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream(); byte[] buffer = new byte[2048]; for (int count=0; ( inputStream.read(buffer) ) != -1; ) { arrayOutputStream.write(buffer, 0, count); } ByteArrayInputStream byteArrayOutputStream = new ByteArrayInputStream(arrayOutputStream.toByteArray()); // byteArrayOutputStream 对象就可以作为我们需要复用的数据流,配合 `mark`和`reset` 功能提高io性能。