`
该用户名已经存在
  • 浏览: 308345 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

通过mark和reset方法重复利用InputStream

    博客分类:
  • Java
阅读更多
这篇博客中我们已经简单的知道可以通过缓存InputStream来重复利用一个InputStream,但是这种方式的缺点也是明显的,就是要缓存一整个InputStream内存压力可能是比较大的。如果第一次读取InputStream是用来判断文件流类型,文件编码等用的,往往不需要所有的InputStream的数据,或许只需要前n个字节,这样一来,缓存一整个InputStream实际上也是一种浪费。

其实InputStream本身提供了三个接口:
第一个,InputStream是否支持mark,默认不支持。
public boolean markSupported() {
   return false;
}

第二个,mark接口。该接口在InputStream中默认实现不做任何事情。
public synchronized void mark(int readlimit) {}

第三个,reset接口。该接口在InputStream中实现,调用就会抛异常。
public synchronized void reset() throws IOException {
   throw new IOException("mark/reset not supported");
}

从三个接口定义中可以看出,首先InputStream默认是不支持mark的,子类需要支持mark必须重写这三个方法。
第一个接口很简单,就是标明该InputStream是否支持mark。
mark接口的官方文档解释:
“在此输入流中标记当前的位置。对 reset 方法的后续调用会在最后标记的位置重新定位此流,以便后续读取重新读取相同的字节。
readlimit 参数告知此输入流在标记位置失效之前允许读取许多字节。

mark 的常规协定是:如果方法 markSupported 返回 true,则输入流总会在调用 mark 之后记住所有读取的字节,并且无论何时调用方法 reset ,都会准备再次提供那些相同的字节。但是,如果在调用 reset 之前可以从流中读取多于 readlimit 的字节,则根本不需要该流记住任何数据。”


reset接口的官方文档解释:

将此流重新定位到对此输入流最后调用 mark 方法时的位置。
reset 的常规协定是:


如果方法 markSupported 返回 true,则:
如果创建流以来未调用方法 mark,或最后调用 mark 以来从该流读取的字节数大于最后调用 mark 时的参数,则可能抛出 IOException。
如果未抛出这样的 IOException,则将该流重新设置为这种状态:最近调用 mark 以来(或如果未调用 mark,则从文件开始以来)读取的所有字节将重新提供给 read 方法的后续调用方,后接可能是调用 reset 时的下一输入数据的所有字节。
如果方法 markSupported 返回 false,则:
对 reset 的调用可能抛出 IOException。
如果未抛出 IOException,则将该流重新设置为一种固定状态,该状态取决于输入流的特定类型和其创建方式的固定状态。提供给 read 方法的后续调用方的字节取决于特定类型的输入流。



简而言之就是:
调用mark方法会记下当前调用mark方法的时刻,InputStream被读到的位置。
调用reset方法就会回到该位置。
举个简单的例子:
String content = "BoyceZhang!";
InputStream inputStream = new ByteArrayInputStream(content.getBytes());

// 判断该输入流是否支持mark操作
if (!inputStream.markSupported()) {
    System.out.println("mark/reset not supported!");
}
int ch;  
boolean marked = false;  
while ((ch = inputStream.read()) != -1) {
    
    //读取一个字符输出一个字符  
    System.out.print((char)ch);  
    //读到 'e'的时候标记一下
     if (((char)ch == 'e')& !marked) {  
        inputStream.mark(content.length());  //先不要理会mark的参数
         marked = true;  
     }  
                
     //读到'!'的时候重新回到标记位置开始读
      if ((char)ch == '!' && marked) {  
          inputStream.reset();  
          marked = false;
      }  
}

//程序最终输出:BoyceZhang!Zhang!

看了这个例子之后对mark和reset接口有了很直观的认识。
但是mark接口的参数readlimit究竟是干嘛的呢?
我们知道InputStream是不支持mark的。要想支持mark子类必须重写这三个方法,我想说的是不同的实现子类,mark的参数readlimit作用不尽相同。
常用的FileInputStream不支持mark。
1. 对于BufferedInputStream,readlimit表示:InputStream调用mark方法的时刻起,在读取readlimit个字节之前,标记的该位置是有效的。如果读取的字节数大于readlimit,可能标记的位置会失效。

在BufferedInputStream的read方法源码中有这么一段:
} else if (buffer.length >= marklimit) {
     markpos = -1;   /* buffer got too big, invalidate mark */
     pos = 0;        /* drop buffer contents */
     } else {            /* grow buffer */

为什么是可能会失效呢?
因为BufferedInputStream读取不是一个字节一个字节读取的,是一个字节数组一个字节数组读取的。
例如,readlimit=35,第一次比较的时候buffer.length=0(没开始读)<readlimit
然后buffer数组一次读取48个字节。这时的read方法只会简单的挨个返回buffer数组中的字节,不会做这次比较。直到读到buffer数组最后一个字节(第48个)后,才重新再次比较。这时如果我们读到buffer中第47个字节就reset。mark仍然是有效的。虽然47>35。

2. 对于InputStream的另外一个实现类:ByteArrayInputStream,我们发现readlimit参数根本就没有用,调用mark方法的时候写多少都无所谓。
public void mark(int readAheadLimit) {
   mark = pos;
}

public synchronized void reset() {
   pos = mark;
}


因为对于ByteArrayInputStream来说,都是通过字节数组创建的,内部本身就保存了整个字节数组,mark只是标记一下数组下标位置,根本不用担心mark会创建太大的buffer字节数组缓存。

3. 其他的InputStream子类没有去总结。原理都是一样的。

所以由于mark和reset方法配合可以记录并回到我们标记的流的位置重新读流,很大一部分就可以解决我们的某些重复读的需要。
这种方式的优点很明显:不用缓存整个InputStream数据。对于ByteArrayInputStream甚至没有任何的内存开销。
当然这种方式也有缺点:就是需要通过干扰InputStream的读取细节,也相对比较复杂。
5
2
分享到:
评论

相关推荐

    springboot 解决InputStream只能读取一次的问题

    // 实现其他ServletInputStream的方法,如 available(), markSupported(), mark(int), reset() } } ``` 在这个例子中,我们创建了一个名为`RepeatableHttpServletRequest`的类,它继承自`...

    Java中InputStream类.pdf

    在处理这些数据流时,可以利用`InputStream`提供的方法进行读取、跳过、标记和重置操作。 总结,`InputStream`在Java I/O体系中扮演着至关重要的角色,它是所有字节输入流的基础,通过它的子类,我们可以灵活地处理...

    IO流文档InputStream / OutputStream

    在IO流中, mark()和reset()方法用于标记流中的位置,skip()方法用于跳过流中的某些字节,而available()方法用于获取流中的可用字节数。close()方法用于关闭流,释放系统资源。 在Java中,IO流还可以用于实现对象...

    Java中的字节流文件读取教程(一)

    这些方法都是用于读取文件的字节流, skip方法用于跳过n个字节,close方法用于关闭流并释放对应的资源,mark和reset方法用于标志当前流读取位置和重置读取指针。 5. OutputStream和InputStream的相似性 ...

    BufferedInputStream(缓冲输入流)详解_动力节点Java学院整理

    它的主要作用是为另一个输入流添加一些功能,例如提供“缓冲功能”和支持“mark() 标记”和“reset() 重置方法”。 BufferedInputStream 的工作原理是通过一个内部缓冲区数组实现的。例如,在新建某输入流对应的 ...

    Java_输入输出流及文件读写详解.docx

    如果需要重复读取流中同一段内容,则需要使用流类中的 mark 方法进行标记,然后才能重复读取。 本文档对 Java 中的输入输出流和文件读写进行了详细的介绍,涵盖了 I/O 类的体系结构、字节流和字符流、InputStream ...

    Java字节流 .pdf

    * boolean markSupported():测试此输入流是否支持mark和reset方法。 * abstract int read():从输入流中读取数据的下一个字节。 * int read(byte[] b):从输入流中读取一定数量的字节,并将其存储在缓冲区数组b中。...

    restream:用于缓冲和重新读取 java.io.InputStream 的实用程序类

    在Java编程中,`InputStream`是处理字节流的基础类,广泛用于读取各种输入数据,如文件、网络连接等。...在具体使用时,应仔细阅读库的文档,了解各个类和方法的具体用法,以便更好地融入到自己的项目中。

    java输入输出流总结.docx

    Java 输入输出流是Java IO API的核心部分,用于在...了解和掌握这些基本概念和方法,对于开发高效且可靠的Java应用程序至关重要。在实际编程中,我们还需要注意处理可能出现的IOException等异常,确保程序的健壮性。

    JAVA语言程序设计第七章(2)ppt课件.ppt

    两者都支持close方法以关闭流,还有skip、mark和reset方法用于控制流的读取位置。 Java的流类形成了层次结构,每个核心流类都有其对应的字节流和字符流版本。例如,InputStreamReader和OutputStreamWriter是连接...

    java_IO流的处理.ppt

    Java IO流处理是Java编程中一个非常重要的概念,它用于在不同数据源之间传输数据,如文件、网络、内存等。在Java中,IO流分为两...通过选择合适的流类型和组合使用过滤流,开发者可以实现高效且易维护的数据处理代码。

    Java_输入输出流及文件读写详解

    `mark()`和`reset()`方法支持标记读取位置,便于回溯。`markSupported()`检查流是否支持标记功能。 2. 字节输出流(Byte Output Stream)的基类是`OutputStream`,例如`FileOutputStream`和`BufferedOutputStream`...

    Java流(文件读写操作)

    缓冲流的一个重要特性是可以使用`mark()`和`reset()`方法来标记和重置读取位置。此外,对于输出缓冲流来说,数据会先在内存中缓存,当调用`flush()`方法时,才会将缓存的数据立即写入。 #### 三、类层次结构 Java...

    java输入输出流

    它可以使用`readLine()`方法逐行读取文本,还可以通过`mark()`和`reset()`方法实现流的标记和回溯。 在给出的`IOStreamDemo`示例中,展示了不同类型的输入输出流的使用: 1. 使用`BufferedReader`从键盘读取一行...

    java输入输出流总结.pdf

    Java 输入输出流是Java编程语言中用于处理数据输入和输出的核心机制。在Java中,输入输出流被设计成一种抽象的概念,允许程序处理各种不同类型的输入...理解并熟练运用这些类和方法是编写高效、健壮的Java程序的关键。

    IO体系.java

    | 为另一个输入流添加一些功能,创建一个缓冲区,支持mark和reset方法。 | mark操作记录输入流中的某个点,reset操作在从包含的输入流获取新字节前,再读取最后一次。 |--DataInputStream/:用于操作基本数据...

    计算机软件-商业源码-228 使用流文件读取数据.zip

    8. **高级流操作**:可能涉及到了缓冲区的大小设置,以及`mark()`和`reset()`方法进行数据回溯。 9. **文件复制**:使用流进行文件复制是常见的练习,源码可能包含这一部分,演示了如何从一个文件读取数据,然后...

    java数据流 I/O系统

    Java还提供了针对文件操作的特殊数据流,如`FileInputStream`和`FileOutputStream`,它们可以直接与文件交互,但不支持`mark()`和`reset()`方法。使用这些类时,可以直接传入文件名来创建实例,从而实现对本地文件的...

    JAVA语言课时授课计划10-5页.pdf

    - InputStream的read()方法用于读取单个字节,skip()方法跳过指定字节数,mark()和reset()方法用于设置和回溯读取位置,close()方法用于关闭流。 - OutputStream的write()方法用于写入单个字节,flush()方法刷新...

    语言程序设计资料:java输入输出-(数据库技术-安全技术-).ppt

    - `mark()`:标记流中的当前位置,以便稍后使用`reset()`返回此位置。 - `markSupported()`:检查流是否支持`mark()`和`reset()`操作。 - `read()`:读取单个字节。 - `read(byte[])`和`read(byte[], int, int)`:将...

Global site tag (gtag.js) - Google Analytics