Java 的输入流主要由InputStream 和Reader 作为基类,而输出流则主要由OutputStream 和 writer 作为基类。
在前面的Java IO:概述中,我们已提到过,流分为字节流和字符流、节点流和处理流。
字节流和字符流
区别在于操作的数据单元不同,字节流操作最小数据单元是8位的字节,字符流操作的最小数据单元是16位的字符。
字节流主要是由InputStream和OutputStream 作为基类,而字符流主要由Reader 和 writer作为基类。
节点流和处理流
从/向一个特定的IO设备(磁盘、网络)读/写数据的流,成为节点流。 如:
对一个已存在的流进行封装或连接,通过封装后流来实现数据读/写功能,称为处理流。
注意:当使用处理流来进行输入/输出时,程序并不会直接连接到实际的数据源。
好处:使用处理流包装节点流,即可以消除不同节点流的实现差异,也可以提供方便的方法完成输入/输出功能。
InputStream 和Reader 都是抽象类,本身不能创建实例,但分别有一个用于读取文件的输入流: FileInputStream 和FileInputReader ,它们都是节点流,即直接和指定文件关联。
InputStream 例子
/**
* java input stream
* @author mingx
*
*/
public class InputStreamExample {
public static void main(String[] args) {
int count = 0; // 用于保存实际读取的字节数
InputStream inputStream = null;
try {
inputStream = new FileInputStream(new File("D:\\ZhyTestSpace\\testBase\\src\\test\\java\\testBase\\FileExample.java"));
byte[] bbuf = new byte[1024];
while ((count = inputStream.read(bbuf)) > 0 ) { //读取文件字节
System.out.println("取出" + new String(bbuf,0,count));
}
} catch (final IOException e) {
e.printStackTrace();
} finally {
try {
inputStream.close(); // fileInputStream是有缓冲区的,所以用完之后必须关闭,否则可能导致内存占满,数据丢失。
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
FileInputStream 的read() 方法,返回读取到的包含一个字节内容的int变量(0~255),如果read 返回-1,意味着程序已经读到了末尾,此时流内已经没有多余的数据可供读取,此时可以关闭流。
以上程序,创建一个长度为1024字节的字节数组来读取该文件,该文件长度为497字节,所以我们只需要一次read就可以全部读取全部内容。但是如果我们创建较小长度的字节数组,程序运行时可能存在中文乱码问题,例如我们将上面的字节数组定义修改如下:
byte[] bbuf = new byte[10];
输出打印:
执行了多次read()方法,并且中文呈乱码格式。
最后程序需要在Finally块中使用inputStream.close来关闭该文件输入流。
Reader实例:
public class ReaderExample {
public static void main(String[] args) throws IOException {
FileReader fr = null;
try {
// 创建字符输入流
fr = new FileReader("D:\\ZhyTestSpace\\testBase\\src\\test\\java\\testBase\\UserTest.java");
// 创建一个长度为32的字符数组
char[] cbuf = new char[32];
int count = 0;
while ((count = fr.read(cbuf)) > 0) {
// 取出字节,将字符数组转换成字符串输入
System.out.println("取出" + new String(cbuf, 0, count));
}
} catch (FileNotFoundException e) {
e.printStackTrace();
} finally {
if (fr != null) {
fr.close();
}
}
}
}
ReaderExample与前面的InputStream实例无多大区别,这里将字符数组定义32,也就意味着,需要调用多次read()方法才能完全读取输入流的全部数据。
最后:InputStream和 Reder都支持如下方法来移动记录指针:
- void mark (int readAheadLimit) :在记录指针当前位置记录一个标记
- boolean markSupported() :判断输入流是否支持mark操作
- void reset():将此流的记录指针重新定位到上一次记录标记(mark)位置。
- long skip (long n) :记录指针向前移动n个字节/字符