所有的程序都离不开信息的输入和输出。在Java程序中也经常需要对相关文件等数据进行读写操作,这时就需要用到对输入、输出流的相关处理操作。例如,从键盘读取数据、在网络上交换数据、打印报表、读写文件信息等,都要涉及数据输入输出的处理。在面向对象语言中,输入和输出都是通过数据流来实现的。在Java中,处理数据流的类主要被放在java.io包中 。
个人认为,关于流的处理这里,我在学习时就感觉挺难懂的,但它却是很重要,在以后的Java学习中时常会用到,因此,希望初学的朋友在学习时能够掌握牢固。
数据流的基本概念
数据源(Data Sourcc):是指那些能够提供数据的地方,包括键盘、磁盘文件、网络接口等。
数据宿(Data Sink):指能够接收数据的地方,可以是磁盘文件、网络接口以及显示器、打印机等外部设备。(数据宿也可认为是数据传输的目的地)
在国内业界,人们一般不对这两者进行仔细区分,而是统称为“数据源”
数据流
考虑到数据源的多样性,为了更有效地进行数据的输入、输出操作,Java中把不同的数据源与程序之间的数据传输都抽象表述为“流”(stream),以实现相对统一和简单的输入/输出操作方式。传输中的数据就像流水一样,也称为数据流。
I/O数据流的分类方式
按照数据流动的方向,可分为输入流(Input Stream)和输出流(Output Stream)
输入流只能从中读取数据,而不能向其写出数据;
输出流则只能向其写出数据,而不能从中读取数据。
这里站在程序的角度来确定出入方向,即将数据从程序外部传送到程序中谓之“输入”数据,将程序中的数据传送到外部谓之“输出”数据。
根据数据流所关联的是数据源还是其他数据流,可分为节点流(Node Stream)和处理流(Processing Stream)
节点流可以从/向一个特定的地方读/写数据。
“地方”就是数据源,如键盘或磁盘文件等。
即节点流是直接连接到数据源的I/O流
例,FileReader fr=new FileReader(“demo.txt”);
处理流是对一个已存在的流的连接和封装,通过所封装的流的功能调用实现增强的数据读/写功能,处理流并不直接连接到数据源。
例,BufferedReader br=newBufferedReader(new FileReader(“demo.txt”));
按传输数据的“颗粒大小”划分,可分为字符流(Character Stream)和字节流(Byte Stream)
字节流以字节为单位进行数据传输,每次传送一个或多个字节;
字符流以字符为单位进行数据传输,每次传送一个或多个字符;(字符流使用了字节流读到一个或多个字节(中文对应的字节数是两个,在UTF-8码表中是3个字节)时,先去查指定的编码表,将查到的字符返回)
字节流可以处理所有类型的数据,如图片、mp3、avi
字符流只能处理字符数据
结论:只要是处理纯文本数据,就要优先考虑使用字符流,除此之外都用字节流
输入/输出流体系
注意:Java采用的命名惯例有助于我们区分字节流类型和字符流类型:凡是以InputStream或OutputStream结尾的类型均为字节流,凡是以Reader或Writer结尾的均为字符流。(IO体系中的子类名后缀都是父类名称,而前缀都是体现子类功能的名字)。
字符流
Reader类和Writer类用来专门处理字符流。这两个类都是抽象类,从他们派生出了许多子类,增强功能、提高效率,实现各种不同要求的字符输入/输出流的处理。
Reader
抽象类java.io.Reader是所有字符输入流类型的父类,其中声明了用于读取字符流的有关方法,其类层次关系为:
注意:IO体系中的子类名后缀都是父类名称,而前缀都是体现子类功能的名字
Reader类中定义的方法
public intread():读取一个字符,返回的是读到的那个字符。如果读到流的末尾,返回-1。
public intread(char[] cbuf):将读到的字符存入指定的数组中,返回的是实际读取的字符数。如果读到流的末尾,返回-1。
public abstractint read(char[] cbuf,int off,int len):将读到的字符存入数组的指定位置(off),每次最多读len个字符,返回实际读取的字符数。如果读到流的末尾,返回-1。
close():读取字符其实用的是window系统的功能,使用完毕后,进行资源的释放。
Writer
Java.io.Writer与java.io.Reader类对应,是所有字符输出流类型的共同父类,也是一个抽象类,其中声明了用于写字符流的有关方法。
Writer类中定义的主要方法
public voidwrite(int c):将一个字符写入到流中。
public voidwrite(char[]):将数组中的字符依次写出。
public abstractvoid write(char[] bcbuf,int off,int len):将数组中下标off开始的len个字符写出。
public voidwrite(String):将一个字符串写入到流中。
public abstractvoid flush():刷新流,将流中的数据刷新到目的地中,流还存在。
public abstreactvoid close():关闭资源,关闭前会先调用flush,刷新流中的数据去目的地,然后流关闭。
FileWriter的使用
该类没有特有的方法。只有自己的构造函数。
该类特点:
用于处理文本文件。
该类中有默认的编码表,
该类中有临时缓冲。
构造函数
publicFileWriter(String filename);//调用系统资源,在指定位置,创建一个文件。注意:如果该文件已存在,将会被覆盖。
publicFileWriter(String filename,boolean append);//当传入的boolean类型的参数值为true时,会在指定文件末尾处进行数据的续写。
实例1,将文本数据存储到一个文件中。
class Demo{ public static void main(String[]args)throws IOException{ FileWriter fw = newFileWriter("demo.txt"); fw.write("abcdec"); fw.flush(); fw.write("kkkk"); fw.close(); } }
注意:对于读取或者写入流对象的构造函数,以及读写方法,还有刷新关闭功能都会抛出IOException或其子类,所以都要进行处理
实例2,完整的异常处理方式。
class Demo{ publicstatic void main(String[] args)throws IOException{ FileWriterfw = null; try{ fw= new FileWriter("z:\\demo.txt"); fw.write("abcdec"); fw.flush(); fw.write("kkkk"); }catch(IOException e){ System.out.println(e.toString()); }finally{ if(fw!=null) try{ fw.close(); }catch(IOExceptione){ System.out.println("close:"+e.toString()); } } } }
当指定绝对路径时,定义目录分隔符有两种方式:
• 反斜线 但是一定要写两个。\\ new FileWriter("c:\\demo.txt");
• 斜线 / 写一个即可。 newFileWriter("c:/demo.txt");
FileReader的使用
• 用于读取文本文件的流对象。
• 构造函数:在读取流对象初始化的时候,必须要指定一个被读取的文件。
public FileReader(String filename);//如果该文件不存在会发生FileNotFoundException
注意:read方法若返回-1,则表明当前读取位置已经到达流的末尾。例如:
int ch; while((ch=fr.read())!=-1){ //数据处理(char)ch }