InputStream这个抽象类是所有基于字节的输入流的超类,抽象了Java的字节输入模型。在这个类中定义了一些基本的方法。看一下类的定义:
public abstract class InputStream implements Closeable</span>
首先这是一个抽象类,实现了Closeable接口,也Closeable接口又拓展了AutoCloseable接口,因此所有InputStream及其子类都可以用于Java 7 新引入的带资源的try语句。读入字节之前,我们可能想要先知道还有多少数据可用,这有available方法完成,具体的读入由read()及其重载方法完成,skip方法用于跳过某些字节,同时定义了几个有关标记(mark)的方法,读完数据使用close方法关闭流,释放资源。下面详细介绍各个方法:
1. available方法
public int available() throws IOException
假设方法返回的int值为a,a代表的是在不阻塞的情况下,可以读入或者跳过(skip)的字节数。也就是说,在该对象下一次调用读入方法读入a个字节,或者skip方法跳过a个字节时,不会出现阻塞(block)的情况。这个调用可以由相同线程调用,也可以是其他线程调用。但是在下次读入或跳过的时候,实际读入(跳过)的可能不只a字节。当遇到流结尾的时候,返回0。如果出现I/O错误,抛出IOException异常。看一下InputStream中该方法的实现:
public int available() throws IOException {
return 0;
}
只是简单地返回0,因此子类必须重写该方法。注意到一点,虽然这个方法实现中根本不会出现异常,但是还是在throws中指出(specify)可能抛出IOException。这是Java异常机制很重要的一个点,子类的方法不能throws父类方法没有throws的异常(构造器除外),因此在父类方法先指出,然后允许子类方法抛出IOException。详细介绍可以参考我的另一篇博文
《Java异常》。下面看一个例子:假设我们的test.txt中有ABCDE5个字母(没有空格),如下程序将输出可用的字节数(5)。
public static void main(String[] args) throws IOException {
InputStream is = null;
//byte[] buffer = new byte[4];
//char c;
try {
is = new FileInputStream("test.txt");
System.out.println("available: " + is.available());
/*for (byte b : buffer) {
System.out.println((char)b);
}*/
} finally {
if (is != null) {
is.close();
}
}
单独使用这一方法几乎没有意义,它一般用于在读入或者跳过之间先探测一下有多少可用字节。
2. 读入方法:read
跟读入相关的方法是这个类的核心方法。有3种重载的形式,下面分别介绍。
2.1 read()
public abstract int read() throws IOException
读取输入流的下一个字节。这是一个抽象方法,不提供实现,子类必须实现这个方法。该方法读取下一个字节,返回一个0-255之间的int类型整数。如果到达流的末端,返回-1. 调用该方法的时候,方法阻塞直到出现下列其中一种情况:1)遇到流的尾部(end of the stream)。2)有数据可以读入。3)抛出异常。修改一下上面的例子,把try语句块改为:
is = new FileInputStream("test.txt");
int i;
while ((i = is.read()) != -1) {
System.out.println("out: " + (char)i);
}
输出:
out: A
out: B
out: C
out: D
out: E
面向字节的操作时,可能需要像这样比较底层的字节操作。我们也可以一次读入多个字节,使用下面的重载形式。
2.2 read(byte[] b)
public int read(byte b[]) throws IOException
试图读入多个字节,存入字节数组b,返回实际读入的字节数。
如果传递的是一个空数组(注意数组长度可以为0,即空数组。比如 byte[] b = new byte[0]; 或者byte[] b = {};)那么什么也没读入,返回0.
如果到达流尾部,没有字节可读,返回-1;
如果上面两种情况都没有出现,并且没有I/O错误,则至少有1个字节被读入,存储到字节数组b中。实际读入的第一个字节存在b[0],往后一次存入数组,读入的字节数最多不能超过数组b的长度。如果读入的字节数小于b的长度,剩余的数组元素保持不变。具体地,如果读入的字节数为k,则k个字节分别存在b[0]到b[k-1],而b[k]到b[b.length-1]保持原来的数据。还是上面的例子做修改:
InputStream is = null;
byte[] buffer = new byte[4];
//char c;
try {
is = new FileInputStream("test.txt");
is.read(buffer);
//System.out.println("available: " + is.available());
for (byte b : buffer) {
System.out.println((char)b);
}
此时缓存数组buffer长度为4,所以读入测试文件中的前4个字节,填入buffer,输出为:
A
B
C
D
现在我们把字节数组长度改为6,read方法试图读入6个字节,但是文件中只有5个字节,因此读入这5个字节,输出ABCDE。如果此输入buffer的最后一个元素,会发现它还是原来的默认值。如果你不想把读入的数据从第一个元素开始存储,这时可以使用下面的方法。
2.3 read (byte[] b, int off, int len)
public int read(byte[] b,int off,int len) throws IOException
这个方法跟上一个功能类似,除了读入的数据存储到b数组是从off开始。len是试图读入的字节数,返回的是实际读入的字节数。
如果len=0,则什么也不读入,返回0;如果遇到流尾部,返回-1.否则至少读入一个字节。
假设实际读入k个字节,则k个字节分别存储在b[off]到b[off+k-1],而b[off+k]往后的元素保持不变。b[off]之前也保持不变。继续修改:
InputStream is = null;
byte[] buffer = new byte[6];
//char c;
try {
is = new FileInputStream("test.txt");
is.read(buffer, 1, 3);
//System.out.println("available: " + is.available());
for (byte b : buffer) {
System.out.println((char)b);
}
System.out.println((char)buffer[1]);
这时候读入三个字节,分别存入到buffer[1] buffer[2] buffer[3].其他元素保持不变。(注意Java不会为方法中的局部变量进行默认值初始化工作)。
这个方法可能会出现数组越界,比如off>(b.length-1)或者off是负数。或者(off+len)>b.length。
对比上面两个方法,其实read(byte b[])相当于 read(byte[] b , 0, b.length),看一下该方法的源码:
public int read(byte b[]) throws IOException {
return read(b, 0, b.length);
}
解析来看一下最第三个read方法的源代码:
public int read(byte b[], int off, int len) throws IOException {
if (b == null) { // 检测参数是否为null
throw new NullPointerException();
} else if (off < 0 || len < 0 || len > b.length - off) {
throw new IndexOutOfBoundsException(); // 数组越界检测
} else if (len == 0) {
return 0; //如果b为空数组,返回0
}
int c = read(); // 调用read()方法获取下一个字节
if (c == -1) {
return -1;
} // 遇到流尾部,返回-1
b[off] = (byte)c; //读入的第一个字节存入b[off]
int i = 1; // 统计实际读入的字节数
try {
for (; i < len ; i++) { // 循环调用read,直到流尾部
c = read();
if (c == -1) {
break;
}
b[off + i] = (byte)c; // 一次存入字节数组
}
} catch (IOException ee) {
}
return i; // 返回实际读入的字节数
}
我们看到方法specify可能抛出IOException异常,如果第一个字节无法读入且原因不是到达流尾部,或者流已经被关闭,或者其他IO错误,则抛出这个异常。
上面三个读入方法都可能出现阻塞,在2.1中已经介绍了阻塞解除的条件。理解这三个方法很重要的一点是:方法只是尝试读入我们想要的字节数,但是能否成功,会受到数据源的影响。另外一点就是读入的数据存到哪里,第一个方法作为返回值,第二、三个方法存入到指定数组的指定位置,返回的是实际读入的字节数。后两个方法真正的读入工作都是通过调用抽象方法read()来完成的,资格抽象方法在子类中实现。
3. skip方法
public long skip(long n) throws IOException
这个方法试图跳过当前流的n个字节,返回实际跳过的字节数。如果n为负数,返回0.当然子类可能提供不能的处理方式。n只是我们的期望,至于具体跳过几个,则不受我们控制,比如遇到流结尾。修改上面的例子:
InputStream is = null;
byte[] buffer = new byte[6];
//char c;
try {
is = new FileInputStream("test.txt");
is.skip(2);
is.read(buffer, 1, 3);
//System.out.println("available: " + is.available());
for (byte b : buffer) {
System.out.println((char)b);
}
跳过前面2个字节,然后继续读入三个字节。如果你试图跳过6个字节,你会发现实际上值跳过5个字节,而且read方法将返回-1.。形象一点,看一下下面这个图:
跳过2个字节的时候位置由A跳到B,而你如果试图一次跳过6个字节,你会发现它只能跳过5个字节到达C。最后来看一下这个方法在InputStream来中的实现:
public long skip(long n) throws IOException {
long remaining = n; // 还有多少字节没跳过
int nr;
if (n <= 0) {
return 0; // n小于0 简单返回0
}
int size = (int)Math.min(MAX_SKIP_BUFFER_SIZE, remaining); // 这里的常数在类中定义为2048
byte[] skipBuffer = new byte[size]; // 新建一个字节数组,如果n<2048,数组大小为n,否则为2048
while (remaining > 0) {
nr = read(skipBuffer, 0, (int)Math.min(size, remaining)); // 读入字节,存入数组
if (nr < 0) { // 遇到流尾部 跳出循环
break;
}
remaining -= nr;
}
return n - remaining;
}
从代码的逻辑上可以看出,是通过不断地读取字节来完成跳过的任务的。首先建立一个缓冲数组,这个数组的大小不超过2048.如果要跳过的字节大于2048,则数组大小为2048,否则采用要跳过的字节数作为数组长度。接下来不断读取字节,填入数组,如果还没跳过的字节数超过缓冲数组长度,则读入2048,否则读入还没跳过的字节,完成跳过任务。如果遇到流尾部,跳出循环,返回已经读入的字节个数。
4.与标记相关的方法
与标注相关的方法使得我们可以重新读入(跳过)那些已经被我们读入或者跳过的字节,再重新来过一次。
4.1 mark
public void mark(int readlimit)
这个方法用于在流的当前位置做个标记,参数readLimit指定这个标记的“有效期“,如果从标记处开始往后,已经获取或者跳过了readLimit个字节,那么这个标记失效,不允许再重新回到这个位置(通过reset方法)。也就是你想回头不能走得太远呀,浪子回头不一定是岸了,跳过(获取)了太多字节,标记就不再等你啦。多次调用这个方法,前面的标记会被覆盖。
看一下上面的图,如果我们在M出做标记,readLimit为绿色部分,当流的指针在A处的时候,这个标记依然有效,可是一旦指针跑到B处,标记就失效了。
4.2 reset
public void reset() throws IOException
这个方法用于重定位到最近的标记。如果在这之前mark方法从来没被调用,或者标记已经无效,在抛出IOException。如果没有抛出这个异常,将当前位置重新定位到最近的标记位置。
InputStream is = null;
//byte[] buffer = new byte[6];
//char c;
try {
is = new BufferedInputStream(new FileInputStream("test.txt"));
is.mark(4);
is.skip(2);
is.reset();
//is.read(buffer, 1, 3);
System.out.println((char)is.read());
/*for (byte b : buffer) {
System.out.println((char)b);
}*/
} finally {
if (is != null) {
is.close();
}
}
}
我们使用了支持mark的BufferedInputStream,首先一开始做标记,跳过两个自己,然后再回到最初的位置。
4.3 markSupported
public boolean markSupported()
检测当前流对象是否支持标记。是返回true。否则返回false。比如InputStream不支持标记,而BufferedInputStream支持。
5. close方法
public void close() throws IOException
关闭当前流,释放与该流相关的资源,防止资源泄露。在带资源的try语句中将被自动调用。关闭流之后还试图读取字节,会出现IOException异常。
分享到:
相关推荐
我们可以创建一个自定义的`HttpServletRequestWrapper`子类,重写`getInputStream()`方法,使其返回一个可以重复读取的`InputStream`。 以下是一个简单的自定义`HttpServletRequestWrapper`示例: ```java import ...
Java InputStream 的多种使用详解 InputStream 是 Java 中用于读取数据流的抽象类,它是所有输入流的父类。InputStream 的子类有多种,如 FileInputStream、BufferedInputStream、ByteArrayInputStream 等,每种...
【InputStream详解】InputStream是Java IO框架中的核心类,它是所有字节输入流的抽象超类。这个类提供了读取字节的基本操作,并为其他输入流类提供了基础结构。在Java中,任何可以从源头读取字节数据的类都可能是...
Class.getResourceAsStream() 方法返回的是一个 InputStream 对象,该对象可以读取文件的内容。该方法有两个参数,第一个参数是文件的相对路径,第二个参数是 Class 对象。相对路径可以是相对于当前类所在的包的路径...
Java 中的 IO 详解 Java IO 是指应用程序对外部设备的数据输入和输出,在程序中,键盘被当作输入文件,显示器被当作输出文件使用。Java 语言定义了许多类专门负责各种方式的输入输出,这些类都被放在 java.io 包中...
Spring Boot读取resources目录文件方法详解 Spring Boot是一款流行的Java框架,广泛应用于Web应用程序开发中。随着项目的增长,配置文件、模板文件等资源文件的管理变得越来越复杂。因此,学习如何读取resources...
Java IO 详解 Java IO(Input/Output)是 Java 语言中用于处理输入输出操作的类库,提供了大量的类和方法来实现文件、网络、字节流等方面的输入输出操作。下面对 Java IO 中的重要知识点进行详细说明。 一、File ...
InputStream 类定义了一套所有字节输入流所需的方法,包括 read()、skip()、available()、mark()、reset()、markSupported() 和 close() 等。OutputStrean 类定义了一套所有字节输出流所需的方法,包括 write()、...
- 不涉及文件内容的实际读写操作,这部分工作通常通过其他类来完成,如`InputStream`和`OutputStream`。 - **File类的构造方法**: - `File(String directoryPath)`:根据给定的路径创建一个`File`对象。 - `...
**SharePreference与File详解** 在Android开发中,数据存储是一个重要的环节,特别是在处理用户设置或者应用内部状态时。本文将详细解析两种常见的本地数据存储方式:`SharePreference`和`File`。 ### 1. ...
Java_IO流详解 Java 的核心库 java.io 提供了全面的 IO 接口,包括文件读写、标准设备输出等。Java 中 IO 是以流为基础进行输入输出的,所有数据被串行化写入输出流,或者从输入流读入。在项目开发中,IO 是非常...
该类提供了一个 readInputStream 方法,用于读取输入流,并将其转换为字节数组。 四、WebService 服务类 WebService 服务类是一个用来获取图片数据的服务类。它提供了一个 getImage 方法,用于从网络获取图片数据...
本文实例讲述了Android通过json向MySQL中写入数据的方法。分享给大家供大家参考,...static InputStream is = null; static JSONObject jObj = null; static String json = ; // constructor public JSONParser() { }
5. **输入/输出流**:Java的I/O流系统支持文件读写、网络通信等多种数据传输方式,书中会介绍InputStream、OutputStream、Reader、Writer以及NIO(非阻塞I/O)的相关知识。 6. **多线程**:Java2对多线程编程的支持...
java IO部分详解各种流的解析例如 InputStream
InputStream 类中包含的每个方法都会被所有字节输入流类继承,使每个子类根据需要覆盖对应的方法。这样的设计可以保证每个字节输入流子类在进行实际使用时,开放给程序员使用的功能方法是一致的。 InputStream 类中...
Java使用I/O流读取文件内容的方法详解 Java使用I/O流读取文件内容的方法是Java编程中常用的文件读取方式。该方法可以将文件内容读取到程序中,并对其进行处理和分析。在Java中,I/O流是指数据流向某个对象的数据...
InputStream和OutputStream是所有输入和输出流的基类,BufferedInputStream和BufferedOutputStream用于提高性能,FileInputStream和FileOutputStream用于文件操作,而SocketInputStream和SocketOutputStream则用于...
本部分的io流详解内容主要涵盖以下几个方面: 1. **流的分类**: - 根据数据传输方向,流可以分为输入流(Input Stream)和输出流(Output Stream)。 - 根据处理的数据类型,流分为字节流(Byte Stream)和字符...