自学习Java IO以来,认为难倒是不难,就是比较繁杂,各种方式操作的IO类多如麻,每次使用,都不得不花费大把时间网上溜达一圈,才能完成手头任务。按说,像这些常用的IO操作,对于Java程序员来说,应该达到信手拈来,拿来就用的熟练程度。但我就是记也不住哇,究其原因,就是对Java IO的整体结构不甚了解。结构含糊,就不容易记忆。为了以后方便,于是有了整理一下Java IO的想法,打算简明易懂的记录Java IO的各种分门别类,争取理清思路,常回来看看,印记于脑海,再用时再也不必花费大量时间。不想从2010年2月3日创建该笔记始,一直拖到今天2011-07-13才算有了眉目,但是还不够完整,需要后期继续补充。
另, 总结代码形成的工具类请参见 文件操作工具类
一、基本概念
1. 概述
使用IO时,首先创建一个数据源IO,然后根据需要,创建装饰类IO。Java IO 相关的类总结如下表:
字节流 | 字符流 | ||||
|
输入 |
输出 |
输入 |
输出 |
|
数据源IO | |||||
1 | FileInputStream | FileOutputStream | FileReader | FileWriter | |
2 | ByteArrayInputStream | ByteArrayOutputStream | CharArrayReader | CharArrayWriter | |
3 | StringBufferInputStream | |
StringReader | StringWriter | |
4 | SequenceInputStream | SequenceOutputStream | |
|
|
5 | PipedInputStream | PipedOutputStream | PipedReader | PipedWriter | |
6 | System.in | |
|
|
|
装饰类IO | |||||
7 | DataInputStream | DataOutputStream | |
|
|
8 | BufferedInputStream | BufferedOutputStream | BufferedReader | BufferedWriter | |
9 | LineNumberInputStream | |
LineNumberReader | ||
10 | 产生格式化输出PrintStream | 产生格式化输出PrintWriter | |||
11 | PushbackInputStream 一般用于编译器开发 | PushBackReader |
说明:
2. Java IO的使用原则
决定使用哪个类以及它的构造进程的一般准则如下(不考虑特殊需要)
第一,考虑最原始的数据格式是什么:是否为文本?
第二,是输入还是输出?
第三,是否需要转换流:InputStreamReader, OutputStreamWriter?
第四,数据来源(去向)是什么:文件?内存?网络?
第五,是否要缓冲:bufferedReader (特别注明:一定要注意的是readLine()是否有定义,有什么比read, write更特殊的输入或输出方法)
第六,是否要格式化输出:print?
(1) 按数据来源(去向)分类
- 是文件: FileInputStream, FileOutputStream, FileReader, FileWriter
- 是byte[]:ByteArrayInputStream, ByteArrayOutputStream
- 是Char[]: CharArrayReader, CharArrayWriter
- 是String: StringBufferInputStream, StringReader, StringWriter
- 网络数据流:InputStream, OutputStream, Reader, Writer
(2) 按是否格式化输出分
- 要格式化输出:PrintStream, PrintWriter
(3) 按是否要缓冲分
- 要缓冲:BufferedInputStream, BufferedOutputStream, BufferedReader, BufferedWriter
(4) 按数据格式分
- 二进制格式(只要不能确定是纯文本的): InputStream, OutputStream及其所有带Stream结束的子类
- 纯文本格式(含纯英文与汉字或其他编码方式);Reader, Writer及其所有带Reader, Writer的子类
(5) 按输入输出分
- 输入:Reader, InputStream类型的子类
- 输出:Writer, OutputStream类型的子类
(6) 特殊需要
- 从Stream到Reader,Writer的转换类:InputStreamReader, OutputStreamWriter
- 对象输入输出:ObjectInputStream, ObjectOutputStream
- 进程间通信:PipeInputStream, PipeOutputStream, PipeReader, PipeWriter
- 合并输入:SequenceInputStream
- 更特殊的需要:PushbackInputStream, PushbackReader, LineNumberInputStream, LineNumberReader
二、 例子代码
1.文件操作
(1) 从文件读
- 字节流
FileInputStream 的read方法 : throws IOException
int read()读取单个字节
int read(byte[] b)从输入流中将最多 b.length 个字节的数据读入到 byte 数组中
b - 存储读取数据的缓冲区。
返回:读入缓冲区的字节总数,如果到达文件末尾而没有更多的数据,返回 -1
int read(byte[] b, int off,int len)读取len个字节到字节数组b中的指定位置
b - 目标缓冲区。
off - 开始存储字节处的偏移量。
len - 要读取的最大字节数。
//1. 以字节为单位读取文件内容,一次读一个字节 FileInputStream in = new FileInputStream(file); int temp; while ((temp = in.read()) != -1) {//将读取的字节赋值给temp System.out.write(temp); } in.close(); //2. 以字节为单位读取文件内容,一次读多个字节 int byteread = 0; FileInputStream in = new FileInputStream(fileName); byte[] tempbytes = new byte[in.available()];// 一次读多个字节 // 读入多个字节到字节数组中,byteread为一次读入的字节数 while ((byteread = in.read(tempbytes)) != -1) { // String s=new String(tempbytes,"UTF-8"); System.out.write(tempbytes, 0, byteread); }
//3. 带缓冲的读文件 BufferedInputStream 只有read方法,没有readLine方法
BufferedInputStream的read方法 :throws IOException
int read()读取单个字节
int read(byte[] b)从输入流中将最多 b.length 个字节的数据读入到 byte 数组中
b - 存储读取数据的缓冲区。
返回:读入缓冲区的字节总数,如果到达文件末尾而没有更多的数据,返回 -1
int read(byte[] b, int off,int len)读取len个字节到字节数组b中的指定位置
b - 目标缓冲区。
off - 开始存储字节处的偏移量。
len - 要读取的最大字节数。
byte[] data = new byte[10]; BufferedInputStream bis = new BufferedInputStream( new FileInputStream("C:\\readFile.txt")); BufferedOutputStream bos = new BufferedOutputStream( new FileOutputStream("C:\\saveFile.txt")); while (bis .read(data) != -1) bos.write(data); // 将缓冲区中的资料全部写出 bos.flush();
- 字符流
FileReader的read方法:
int read()读取单个字节
int read(byte[] b)从输入流中将最多 b.length 个字节的数据读入到 byte 数组中
b - 存储读取数据的缓冲区。
返回:读入缓冲区的字节总数,如果到达文件末尾而没有更多的数据,返回 -1
int read(byte[] b, int off,int len)读取len个字节到字节数组b中的指定位置
b - 目标缓冲区。
off - 开始存储字节处的偏移量。
len - 要读取的最大字节数。
//1.一次读多个字符 read(byte[] b) char data[] = new char[10];//建立可容纳10个字符的数组 FileReader fr = new FileReader("algorithms.xml"); int num = fr.read(data);// 将数据读入字符列表data内 String str = new String(data, 0, num);// 将字符列表转换成字符串 fr.close(); //2.一次读多个字符 read(byte[] b, int off,int len) File fileName=new File("d:\\temp\\save.txt"); char data[] = new char[(int)fileName.length()];//根据文件长度建立数组 FileReader fr = new FileReader(fileName); int num = fr.read(data,4,6);//从文件中读6个字符放到data数组中,起始位置为data[4] //返回值num为从文件中读取字符的长度,即6 String str = new String(data, 0, num);// 将字符列表转换成字符串 fr.close();
//3. 带缓冲的读文件
BufferedReader的read方法: 增加了一个readLine() 方法throws IOException
int read() 读取单个字符。
int read(char[] cbuf, int off, int len) 将字符读入数组的某一部分。
cbuf - 目标缓冲区
off - 开始存储字符处的偏移量
len - 要读取的最大字符数
String readLine() 读取一个文本行。
//以行为单位从一个文件读取数据, BufferedReader 有read方法和readLine方法 BufferedReader in = new BufferedReader( new FileReader("F:\\TestIO.java")); String s=null; while((s = in.readLine()) != null) { s+="\n"; System.out.print(s); } in.close();
(2) 写文件
- 字节流
FileOutputStream的write方法: throws IOException
void write(int b)将指定字节b写入文件
b - 要写入的字节
void write(byte[] b)将b.length 个字节从指定 byte 数组写入此文件输出流中
void write(byte[] b,int off,int len)将byte数组中从偏移off开始的len个字节写入文件输出流
b - 数据
off - 数据中的起始偏移量
len - 要写入的字节数
//1. 不带缓冲的写文件 byte b[]=new byte[512]; int count=System.in.read(b); //从键盘读取一行字符(写一个读一个,最大读取512个,回车读取结束),存储到缓冲区 boolean a = true;//true表示追加,false表示覆盖 FileOutputStream fos = new FileOutputStream("C:\\saveFile.txt", a); fos.write(b);//把缓冲区b的内容写到指定的文件中 fos.close();
//2. 带缓冲的写文件 见文件操作(读)
BufferedOutputStream的方法: throws IOException
void write(int b)将指定字节b写入文件
b - 要写入的字节
void write(byte[] b)将b.length 个字节从指定 byte 数组写入此文件输出流中
void write(byte[] b,int off,int len)将byte数组中从偏移off开始的len个字节写入文件输出流
b - 数据
off - 数据中的起始偏移量
len - 要写入的字节数
void flush()刷新此缓冲的输出流,迫使所有缓冲的输出字节被写出到底层输出流中
- 字符流
//1. 不带缓冲的写文件
FileWriter的write方法: throws IOException
void write(int b)将指定字节b写入文件
b - 要写入的字节
void write(byte[] b)将b.length 个字节从指定 byte 数组写入此文件输出流中
void write(byte[] b,int off,int len)将byte数组中从偏移off开始的len个字节写入文件输出流
b - 数据
off - 数据中的起始偏移量
len - 要写入的字节数
FileReader fr = null; FileWriter fw = null; int bi; fr = new FileReader("d:\\temp\\read.txt"); fw = new FileWriter("d:\\temp\\save1.txt"); while ((bi = fr.read()) != -1) { fw.write(bi);//一次读一个字符(一个汉字字符占两个字节,一个英文字符占一个字节) } fr.close(); fw.close();
//2. 带缓冲的写文件
BufferedWriter的方法: throws IOException
void write(char[] cbuf, int off, int len) 写入字符数组的某一部分。
void write(int c) 写入单个字符。
void write(String s, int off, int len) 写入字符串的某一部分
s - 要写入的字符串
off - 开始读取字符处的偏移量
len - 要写入的字符数
void newLine() 写入一个行分隔符。行分隔符字符串由系统属性 line.separator 定义,并且不一定是单个新行 ('\n') 符
void flush()刷新此缓冲的输出流,迫使所有缓冲的输出字节被写出到底层输出流中
public class BufferedReaderWriterDemo { public static void main(String[] args) { try { // 缓冲System.in输入流 // System.in是位流,可以通过InputStreamReader将其转换为字符流 BufferedReader bufReader = new BufferedReader( new InputStreamReader(System.in)); // 缓冲FileWriter BufferedWriter bufWriter = new BufferedWriter(new FileWriter( "F:\\saveFile.txt")); String input = null; // 每读一行进行一次写入动作 while (!(input = bufReader.readLine()).equals("quit")) { bufWriter.write(input); // newLine()方法写入与操作系统相依的换行字符,依执行环境当时的OS来决定该输出那种换行字符 bufWriter.newLine(); } bufReader.close(); bufWriter.close(); } catch (ArrayIndexOutOfBoundsException e) { System.out.println("没有指定文件"); } catch (IOException e) { e.printStackTrace(); } } }
2. 内存读写
ByteArrayInputStream和ByteArrayOutputStream是将位数组当作流输入来源、输出目的地的类。
CharArrayReader和CharArrayWriter将字符数组当作字符数据输入输出的来源和目的
ByteArrayInputStream的方法: throws IOException
构造函数:ByteArrayInputStream(byte[] buf)
创建一个 ByteArrayInputStream,使用 buf 作为其缓冲区数组。
ByteArrayInputStream(byte[] buf, int offset, int length)
创建 ByteArrayInputStream,使用 buf 作为其缓冲区数组。
buf - 输入缓冲区。
offset - 缓冲区中要读取的第一个字节的偏移量。
length - 从缓冲区中读取的最大字节数。
int read()读取单个字节
int read(byte[] b)从输入流中将最多 b.length 个字节的数据读入到 byte 数组中
b - 存储读取数据的缓冲区。
返回:读入缓冲区的字节总数,如果到达文件末尾而没有更多的数据,返回 -1
int read(byte[] b, int off,int len)读取len个字节到字节数组b中的指定位置
b - 目标缓冲区。
off - 开始存储字节处的偏移量。
len - 要读取的最大字节数。
ByteArrayOutputStream的方法:
byte[] toByteArray()
创建一个新的 byte 数组,大小是此输出流的当前大小,并且缓冲区的有效内容已复制到该数组中。
String toString() 使用平台默认的字符集,通过解码字节将缓冲区内容转换为字符串。
String toString(String charsetName)
使用指定的 charsetName,通过解码字节将缓冲区内容转换为字符串。
void write(byte[] b, int off, int len)
将byte数组中从偏移量off开始的len个字节写入此byte数组输出流。
b - 数据。
off - 数据的初始偏移量。
len - 要写入的字节数。
void write(int b) 将指定的字节写入此 byte 数组输出流。
void writeTo(OutputStream out) 将此 byte 数组输出流的全部内容写入到指定的输出流参数中。
public class ByteArrayStreamDemo { public static void main(String[] args) { try { BufferedInputStream bufInputStream = new BufferedInputStream( new FileInputStream("F:\\read.txt")); ByteArrayOutputStream arrayOutputStream = new ByteArrayOutputStream(); byte[] bytes = new byte[1]; // 将文件内容写入位数组流 while (bufInputStream.read(bytes) != -1) { arrayOutputStream.write(bytes); } arrayOutputStream.close(); bufInputStream.close(); // toByteArray()方法,以字符方式显示位数组内容 bytes = arrayOutputStream.toByteArray(); for (int i = 0; i < bytes.length; i++) { System.out.print((char) bytes[i]); } //toString()方法 System.out.println(arrayOutputStream.toString()); // writeTo方法 //写到控制台输出流(能直接输出) arrayOutputStream.writeTo(System.out); FileOutputStream fo=new FileOutputStream("F:\\saveFile.txt",true); //写到文件输出流,能直接输出 arrayOutputStream.writeTo(fo); // 让使用者输入位置与字符修改位数组内容 Scanner scanner = new Scanner(System.in); System.out.print("请输入修改位置:"); int pos = scanner.nextInt(); System.out.println("输入修改字符"); // 修改数组中对应的字符 bytes[pos - 1] = (byte) scanner.next().charAt(0); // 将位数组内容存回文件 ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream( bytes); // 参数是数组 BufferedOutputStream bufOutputStream = new BufferedOutputStream( new FileOutputStream("F:\\read.txt")); byte[] tmp = new byte[1]; while (byteArrayInputStream.read(tmp) != -1)//从字节数组中读取数据 bufOutputStream.write(tmp); byteArrayInputStream.close(); bufOutputStream.flush(); bufOutputStream.close(); } catch (ArrayIndexOutOfBoundsException e) { System.out.println("请指定文件名称"); } catch (IOException e) { e.printStackTrace(); } } }
CharArrayReader的方法:
构造函数:CharArrayReader(char[] buf)创建一个 CharArrayReader,使用 buf 作为其缓冲区数组。
CharArrayReader (char[] buf, int offset, int length)
创建 CharArrayReader,使用 buf 作为其缓冲区数组。
buf - 输入缓冲区。
offset - 缓冲区中要读取的第一个char的偏移量。
length - 从缓冲区中读取的最字符数。
int read()读取单个字符
int read(char[] c)将最多 c.length 个字符读入到 char 数组中
c - 存储读取数据的缓冲区。
返回:读取的实际字符数;如果到达流末尾,则返回 -1
int read(char[] c, int off,int len)读取len个字节到字节数组b中的指定位置
c - 目标缓冲区。
off - 开始存储字符的偏移量。
len - 要读取的最大字符数。
CharArrayWriter的方法:
char[] toCharArray() 返回输入数据的副本。
String toString() 将输入的数据转换为字符串。
void write(char[] b, int off, int len)
将char数组中从偏移量off开始的len个字节写入此CharArrayWriter数组输出。
c - 数据。
off - 数据的初始偏移量。
len - 要写入的字符数。
void write(String str, int off, int len) 将字符串的某一部分写入缓冲区。
str -要写入的字符串。
off - 数据的初始偏移量。
len - 要写入的字符数。
void write(int b) 将一个字符写入缓冲区。
void writeTo(Writer out) 将缓冲区的内容写入另一个字符流。
public class CharArrayReaderWriterDemo { public static void main(String[] args) { try { BufferedReader bufInputReader = new BufferedReader(new FileReader( "F:\\read.txt")); CharArrayWriter charArrayWriter = new CharArrayWriter(); char[] array = new char[1]; // 将文件读入字符数组 while (bufInputReader.read(array) != -1) { charArrayWriter.write(array); } charArrayWriter.close(); bufInputReader.close(); // toCharArray()方法 显示字符内容 array = charArrayWriter.toCharArray(); for (int i = 0; i < array.length; i++) { System.out.print(array[i] + " "); } //toString()方法 System.out.println(charArrayWriter.toString()); // writeTo方法 写到文件输出流,能直接输出 FileWriter fw=new FileWriter("F:\\saveFile.txt"); charArrayWriter.writeTo(fw); fw.close(); // 将字符数组内容存回文件 CharArrayReader charArrayReader = new CharArrayReader(array); BufferedWriter bufWriter = new BufferedWriter(new FileWriter("F:\\read.txt")); char[] tmp = new char[1]; while (charArrayReader.read(tmp) != -1) { bufWriter.write(tmp); } charArrayReader.close(); bufWriter.flush(); bufWriter.close(); } catch (ArrayIndexOutOfBoundsException e) { System.out.println("请指定文件名!"); } catch (IOException e) { e.printStackTrace(); } } }
3. 其它
InputStreamReader 和 OutputStreamWriter,用于字节流与字符流之间的转换
InputStreamReader从一个数据源读取字节,并自动将其转换成Unicode字符,除非特别声明
OutputStreamWriter将字符的Unicode编码写到输出流,如果你的使用的不是Unicode字符,OutputStreamWriter会将你的字符编码转换成Unicode编码。
在InputStreamReader和OutputStreamWriter的结尾链接一个BufferedReader和BufferedWriter是一个好主意。记住对BufferedWriter使用flush()方法。
// 读写文件的编码:
InputStreamReader r = new InputStreamReader(new FileInputStream(fileName), "utf-8");
OutputStreamWriter out = new OutputStreamWriter(new FileOutputStream(fileName),"utf-8");