Java IO : InputStream、Reader
2016-07-04 17:35
579 查看
Java 的输入流主要由InputStream 和Reader 作为基类,而输出流则主要由OutputStream 和 writer 作为基类。
在前面的Java IO:概述中,我们已提到过,流分为字节流和字符流、节点流和处理流。
字节流和字符流
区别在于操作的数据单元不同,字节流操作最小数据单元是8位的字节,字符流操作的最小数据单元是16位的字符。
字节流主要是由InputStream和OutputStream 作为基类,而字符流主要由Reader 和 writer作为基类。
节点流和处理流
从/向一个特定的IO设备(磁盘、网络)读/写数据的流,成为节点流。 如:
对一个已存在的流进行封装或连接,通过封装后流来实现数据读/写功能,称为处理流。
注意:当使用处理流来进行输入/输出时,程序并不会直接连接到实际的数据源。
好处:使用处理流包装节点流,即可以消除不同节点流的实现差异,也可以提供方便的方法完成输入/输出功能。
InputStream 和Reader 都是抽象类,本身不能创建实例,但分别有一个用于读取文件的输入流: FileInputStream 和FileInputReader ,它们都是节点流,即直接和指定文件关联。
InputStream 例子
以上程序,创建一个长度为1024字节的字节数组来读取该文件,该文件长度为497字节,所以我们只需要一次read就可以全部读取全部内容。但是如果我们创建较小长度的字节数组,程序运行时可能存在中文乱码问题,例如我们将上面的字节数组定义修改如下:
byte[] bbuf = new byte[10];
输出打印:
执行了多次read()方法,并且中文呈乱码格式。
最后程序需要在Finally块中使用inputStream.close来关闭该文件输入流。
Reader实例:
最后:InputStream和 Reder都支持如下方法来移动记录指针:
void mark (int readAheadLimit) :在记录指针当前位置记录一个标记
boolean markSupported() :判断输入流是否支持mark操作
void reset():将此流的记录指针重新定位到上一次记录标记(mark)位置。
long skip (long n) :记录指针向前移动n个字节/字符
在前面的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个字节/字符
相关文章推荐
- java KMP 字符串匹配算法
- myBatis3与spring整合之SqlSessionFactoryBean
- SpringMVC——Controller和视图之间数据的相互传递
- 使用import简化Spring配置文件
- 设计模式--spring源码中使用策略模式(Strategy Pattern)
- Java设计模式(1)单例模式
- 数据处理---Spring Batch之实践
- Spring事务失效的原因
- java反射
- java线程同步锁
- eclipse快捷键及重置快捷键
- 【FAQ】ubuntu:java opensdk 7 如何升级到 8?
- 用java代码调用名片识别接口示例代码
- REST与Spring mvc(未完成)
- RxJava学习笔记(二)--- 操作符
- java annotation
- java语言基础入门——运算符的介绍
- 一致性哈希算法与Java实现
- 使用欧几里得算法计算两个数的最大公约数
- Java的JSON处理器fastjson使用详解