转载自:http://www.importnew.com/23708.html#comments
Java中I/O操作主要是指使用Java进行输入,输出操作. Java所有的I/O机制都是基于数据流进行输入输出,这些数据流表示了字符或者字节数据的流动序列。当程序需要读取数据的时候,就会建立一个通向数据源的连接,这个数据源可以是文件,内存,或是网络连接。类似的,当程序需要写入数据的时候,就会建立一个通向目的地的连接。
数据流分类:
流序列中的数据既可以是未经加工的原始二进制数据,也可以是经一定编码处理后符合某种格式规定的特定数据。因此Java中的流分为两种: 1) 字节流:数据流中最小的数据单元是字节 2) 字符流:数据流中最小的数据单元是字符, Java中的字符是Unicode编码,一个字符占用两个字节。
在学习I/O操作前,先要清楚文件对象的概念。
创建一个文件对象有三种方式:
1.按照绝对路径: File f1 = new File(“d:/xiaowangFolder”);
2.按照相对路径,相对于工作目录,如果在Eclpse中就是项目目录:File f2 = new File(“xiaowang.txt”);
3.把f1作为父目录创建文件对象:File f3 = new File(f1, “xiaowang.exe”);
对于文件对象,有一些常用的方法:
f.exists(); f.isDirectory(); f.isFile(); f.length(); f.lastModified(); f.renameTo(f2);
f.list(); f.listFiles(); f.getParentFile(); f.mkdir(); f.mkdirs(); f.createNewFile(); f.getParentFile().mkdirs(); f.listRoots(); f.delete();
下面学习I/O操作:
InputStream 为字节输入流,它本身为一个抽象类,必须依靠其子类实现各种功能,此抽象类是表示字节输入流的所有类的超类。 继承自InputStream 的流都是向程序中输入数据的,且数据单位为字节(8bit);
Inputstream类中的常用方法:
public abstract int read( ):读取一个byte的数据,返回值是高位补0的int类型值。若返回值=-1说明没有读取到任何字节读取工作结束。
public int read(byte b[ ]):读取b.length个字节的数据放到b数组中。返回值是读取的字节数。该方法实际上是调用下一个方法实现的
public int read(byte b[ ], int off, int len):从输入流中最多读取len个字节的数据,存放到偏移量为off的b数组中。
public int available( ):返回输入流中可以读取的字节数。注意:若输入阻塞,当前线程将被挂起,如果InputStream对象调用这个方法的话,它只会返回0,这个方法必须由继承InputStream类的子类对象调用才有用,
public long skip(long n):忽略输入流中的n个字节,返回值是实际忽略的字节数, 跳过一些字节来读取
public int close( ) :使用完后,必须对我们打开的流进行关闭。
来看看几种不同的InputStream:
FileInputStream把一个文件作为InputStream,实现对文件的读取操作
ByteArrayInputStream:把内存中的一个缓冲区作为InputStream使用
StringBufferInputStream:把一个String对象作为InputStream
PipedInputStream:实现了pipe的概念,主要在线程中使用
SequenceInputStream:把多个InputStream合并为一个InputStream
OutputStream
OutputStream提供了3个write方法来做数据的输出,这个是和InputStream是相对应的。
public void write(byte b[ ]):将参数b中的字节写到输出流。
public void write(byte b[ ], int off, int len) :将参数b的从偏移量off开始的len个字节写到输出流。
public abstract void write(int b) :先将int转换为byte类型,把低字节写入到输出流中。
public void flush( ) : 将数据缓冲区中数据全部输出,并清空缓冲区。
public void close( ) : 关闭输出流并释放与流相关的系统资源。
几种不同的OutputStream:
ByteArrayOutputStream:把信息存入内存中的一个缓冲区中
FileOutputStream:把信息存入文件中
PipedOutputStream:实现了pipe的概念,主要在线程中使用
SequenceOutputStream:把多个OutStream合并为一个OutStream
Reader和InputStream类似;Writer和OutputStream类似。
如何选择I/O流
确定是输入还是输出
输入:输入流 InputStream Reader
输出:输出流 OutputStream Writer
明确操作的数据对象是否是纯文本
是:字符流 Reader,Writer
否:字节流 InputStream,OutputStream
明确具体的设备。
文件:
读:FileInputStream, FileReader,
写:FileOutputStream,FileWriter
数组:
byte[ ]:ByteArrayInputStream, ByteArrayOutputStream
char[ ]:CharArrayReader, CharArrayWriter
String:
StringBufferInputStream(已过时,因为其只能用于String的每个字符都是8位的字符串), StringReader, StringWriter
Socket流
键盘:用System.in(是一个InputStream对象)读取,用System.out(是一个OutoutStream对象)打印
是否需要转换流
是,就使用转换流,从Stream转化为Reader、Writer:InputStreamReader,OutputStreamWriter
是否需要缓冲提高效率
是就加上Buffered:BufferedInputStream, BufferedOuputStream, BufferedReader, BufferedWriter
是否需要格式化输出
注意:所有的流,无论是输入流还是输出流,使用完毕之后,都应该关闭。 如果不关闭,会产生对资源占用的浪费。 当量比较大的时候,会影响到业务的正常开展。
关闭流一般也有三种方法:
1.在try的作用域里关闭文件输入流,这样做有一个弊端;如果文件不存在,或者读取的时候出现问题而抛出异常,那么就不会执行这一行关闭流的代码,存在巨大的资源占用隐患。 不推荐使用
2.在finally中关闭:这是标准的关闭流的方式
(1). 首先把流的引用声明在try的外面,如果声明在try里面,其作用域无法抵达finally.
(2). 在finally关闭之前,要先判断该引用是否为空。(为什么要判空:读取异常或者其他系统异常引发in=null,此时in并没有获取输入流,因此不需要关闭)
(3). 关闭的时候,需要再一次进行try catch处理
这是标准的严谨的关闭流的方式,但是看上去很繁琐,所以写不重要的或者测试代码的时候,都会采用上面的有隐患try的方式。
3.把流定义在try()里,try,catch或者finally结束的时候,会自动关闭