字符流:
1. Writer 写入字符流的抽象类。子类必须实现的方法仅有 write(char[],int,int)、flush()和close()。但是,多数子类将重写此处定义的一些方法,以提高更高的效率和其它功能
与OutputStream一样,对文件的操作使用FileWriter类完成。
2. Reader用于读取字符流的抽象类。子类必须实现的方法只有read(char[],int,int)和close()。但是,多数子类将重写此处定义的一些方法,以提供更高的效率和其它的功能
使用FileReader类对文件进行操作
在上一小节中,我们使用InputStream的实例化对象in进行in.read(bytes)操作时,有可能文件的大小比bytes要大,一个中文字符占两个字节,而一个英文字符占一个字节,这是就有可能形成每次读取到分界处时(比如第一次读取只读取了某个汉字的第一个字节,下一次读取才会读取该汉字的第二个字节)编码错乱。
这时就引入了字符流的概念。
package com.coding.iostudy;
import java.io.File;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
import java.io.Reader;
import java.io.Writer;
public class CharStreamDemo {
public static void main(String[] args) {
// write();
read();
}
public static void write() {
File file = new File("H:/javaCode/folder/b.txt");
try {
Writer out = new FileWriter(file);
String info = "扣丁学堂,让编程更简单!";
out.write(info);
out.write("\r\n");
out.close();
} catch(IOException e) {
e.printStackTrace();
}
}
public static void read() {
File file = new File("H:/javaCode/folder/b.txt");
try {
Reader in = new FileReader(file);
char[] cs = new char[2];
int len = -1;
StringBuffer sb = new StringBuffer();
while((len = in.read(cs)) != -1) {
System.out.println(len);
sb.append(new String(cs,0,len));
}
in.close();
System.out.println(sb);
} catch(IOException e) {
e.printStackTrace();
}
}
}
总结:如果操作的是文本类型的文件,建议使用字符流;如果是非文本类型的文件(音频,图像,视频等),建议使用字节流
***********************
一个重要的地方:我们在进行单步调试时,发现在执行in.close()方法时内容才会写到文件中,因为在执行out.write(info)时会先输出到缓存中,在执行out.close()方法时会默认调用out.flush()方法刷新缓存,将缓存的内容输出到文件中,当然我们也可以自己显式调用out.flush()去刷新缓存。而字节流中默认条件下没有缓存,直接写入到文件中。
缓存可以提升读写效率。我们可以在字节流中加缓存
***********************
在所有的流操作中,字节流是最基础的。任何基于字节的操作都是正确的,无论是文本文件还是二进制文件,只要不打印出来显示。如果确认流里面只有可打印的字符,包括英文和各种国家的文字,可以考虑字符流。由于编码不同,一个字符可能占用多个字节,比如GBK的汉字就占用2个字节,UTF-8的汉字占用3个字节。所以字符流是根据指定的编码,将1个或多个字节转换为java里面的unicode字符,然后进行操作。
字符操作一般使用Writer和Reader等,字节操作一般使用InputStream和OutputStream以及各种包装类。比如BufferedInputStream和BufferedOutputStream等。