以前写Java读写文件的代码,基本上都是到处拷贝,没有深入研究过。以至于有段时间都搞不清楚,使用完一个File对象时候,要不要close。最近写了一些代码也看了一些文章,现在把掌握的I/O知识梳理一下,以备有序补充扩展。
先放一张图,对Java I/O有个总统的认识。从这张图可以很清楚的看清Java I/O的整体情况。大的方面分两类:字节流和字符流。然后就是输入和输出。
记得当初上学的时候,第一次遇到stream的概念就懵了。什么是流?现在好像也不是很明白。现留着吧,占个坑,等明白了再补。
字符流:一般文本文件中存放都是有明确含义的,可供人阅读理解的字符(char)。使用程序读取文件的就希望可以按照字符逐个读取。
字节流:图片、视频文件中存储的都是二进制的字节(byte)。直观的想法,读取的时候当前是按照byte逐个读取。
不管是文本、还是图书、视频最终在磁盘上的时候都是按照byte存储的。因此,可以想象Java要提供基于字符流的机制,就要处理字节和字符的相互转化,这里又涉及字符集合字符编码的问题。
这个问题看着简单,可是又很长一段时间都没搞清楚。这里的根本问题就是以谁作为参考。参考系定了,input、output就不会搞混了。
Java中的输入输出都是以内存为参考的,即往内存里写,叫输入(input);从内存里读,叫输出(output)。至于为什么以内存为参考,我的理解是内存是与计算(即CPU)强相关的,最终的目录是以计算为核心,也就是以处理问题为核心。
public static void copyByChar()
{
FileReader fileReader = null;
FileWriter fileWriter = null;
try
{
fileReader = new FileReader(DATA_FILE_NAME);
fileWriter = new FileWriter(OUTPUT_FILE_NAME);
int value;
while((value = fileReader.read()) != -1)
{
System.out.println(value);
fileWriter.write(value);
}
}
catch(Exception e)
{
e.printStackTrace();
}
finally
{
try
{
fileReader.close();
fileWriter.close();
}
catch (IOException e)
{
e.printStackTrace();
}
}
}
FileReader和FileWriter的基本使用很简单,上面是简单的实例代码。我以前的疑惑是:创建FileReader对象的时候到底要不要创建File对象?因为创建FileReader对象的时候有下面两张方式:
FileReader fileReaderByStream = new FileReader("d:\\test.txt");
FileReader fileReaderByFile = new FileReader(new File("d:\\test.txt"));
字节流(FileInputStream和FileOutputStream)
public static void copyByByte()
{
FileInputStream inStream = null;
FileOutputStream outStream = null;
try
{
inStream = new FileInputStream(DATA_FILE_NAME);
outStream = new FileOutputStream(OUTPUT_FILE_NAME);
int value;
while((value = inStream.read()) != -1)
{
System.out.print(value);
outStream.write(value);
}
}
catch (Exception e)
{
e.printStackTrace();
}
finally
{
try
{
inStream.close();
outStream.close();
}
catch (IOException e)
{
e.printStackTrace();
}
}
}
FileInputStream和FileOutputStream基本用法也很简单,下面是实例代码。
带缓冲的读写(BufferedReader/BufferWriter/BufferInputStream/BufferedOutputStream)
读写磁盘是比较耗时的操作,试想如果每次读写一个字节或字符都进行一次磁盘I/O,那性能一定低的吓人。比较直观的方法当然是,一次读写一批数据。Buffered就是用来解决这个问题的。
BufferedReader/BufferWriter对应FileReader/FileWriter
BufferedInputStream/BufferedOutputStream对应FileInputStream/FileOutputStream
基本用法,如实例代码。
public static void copyWithBuffer()
{
BufferedReader bufferedReader = null;
BufferedWriter bufferedWriter = null;
try
{
bufferedReader = new BufferedReader(new FileReader(DATA_FILE_NAME));
bufferedWriter = new BufferedWriter(new FileWriter(OUTPUT_FILE_NAME));
int value;
while((value = bufferedReader.read()) != -1)
{
System.out.println(value);
bufferedWriter.write(value);
}
}
catch(Exception e)
{
e.printStackTrace();
}
finally
{
try
{
bufferedReader.close();
bufferedWriter.close();
}
catch (IOException e)
{
e.printStackTrace();
}
}
}
流的关闭:
基本原则,谁申请、谁关闭。
Properties props = new Properties();
try
{
props.load(new FileInputStream("message.properties"));
//omitted.
}
catch (Exception ex) {}
props.load是不会关闭流的,因此应该由流的创建者来关闭。
// create and load default properties
Properties defaultProps = new Properties();
FileInputStream in = new FileInputStream("defaultProperties");
defaultProps.load(in);
in.close();
参考文章:
https://docs.oracle.com/javase/tutorial/essential/io/
http://davidisok.iteye.com/blog/2106489
http://stackoverflow.com/questions/3991577/closing-java-inputstreams