之前学习的字节流字符流,都可以读写文件对数据进行操作,当读写数据量很大时,读取的速度会很慢影响效率,那么Java中提供了一套缓冲流,大大的提高了IO流的读写速度
缓冲流分为字节缓冲流和字符缓冲流
1. 字节缓冲流
字节缓冲输出流BufferedOutputStream
字节缓冲输入流BufferedInputStream
他们的内部包含了一个缓冲区,通过缓冲区读写,就可以提高IO流的读写速度
1.1 BufferedOutputStream
API手册中的定义:通过这个输出流,应用程序可以向底层输出流写入字节,而不必为每写入的每个字节调用底层api
也就是说,在jvm上有一个缓冲区buffer,通过这个流是先全部写入buffer中,再一次性调用系统api写入文件中。而前面的流是写一个字节就会调用一次系统api写入文件。新的流就大大的提升了写入效率!
继承自OutputStream,说明是字节流
1.1.1 构造器
BufferedOutputStream(OutputStream out) //采用默认的buffer空间
BufferedOutputStream(OutputStream out, int size) //可以自定义buffer的字节大小
1.1.2 写字节到文件
public static void bufferedOutputStreamDemo() throws IOException{
FileOutputStream fos = new FileOutputStream("/kluter/temp/buf.txt");
BufferedOutputStream bufOs = new BufferedOutputStream(fos);
bufOs.write(97);
bufOs.write("Hello".getBytes());
bufOs.close();
}
1.2 BufferedInputStream
继承自InputStream接口,说明只能输入字节
先将文件数据一个个字节读取到缓冲区,待全部读完最后才写到程序的变量中
1.2.1 构造方法
BufferedInputStream(InputStream in)
BufferedInputStream(InputStream in, int size)
1.2.2 读取文件
public static void bufferedInputStreamDemo() throws IOException{
FileInputStream fis = new FileInputStream("/kluter/temp/buf.txt");
BufferedInputStream bis = new BufferedInputStream(fis);
byte[] barr = new byte[4];
int n = 0;
while((n = bis.read(barr)) != -1){
out.print(new String(barr, 0, n));
}
bis.close();
}
2. 字符缓冲流
同字节流缓冲区类似,字符流也有缓冲区
2.1 BufferedWriter
2.1.1 构造方法
BufferedWriter(Writer out) //常用
BufferedWriter(Writer out, int sz)
可以传递的Writer有,FileWriter和InputStreamWriter,根据是否需要转换编码表决定
2.1.2 使用BufferedWriter写文件
说明:
字符流在写入文件的时候都不能忘了flush()
newLine()是BufferedWriter的特有方法,其他类都没有。具有平台无关性的换行:
在linux下是\n
在windows下是\r\n
2.2 BufferedReader
2.2.1 构造方法
BufferedReader(Reader in)
BufferedReader(Reader in, int sz)
可以传递的Reader有,FileReader和InputStreamReader,根据是否需要转换编码表决定
2.2.2 使用BufferedReader读文件
BufferedReader有一个子类LineNumberReader
BufferedReader还有一个其他类没有的方法String readLine() 读取文本中的一行,以换行符作为区分的一行
3 文件拷贝效率对比
使用248M的大文件测试
3.1 字节流单字节拷贝
3.2 字节流1024数组拷贝
3.3 字节流缓冲区单字节拷贝
3.4 字节流缓冲区1024数组拷贝
3.5 字符流缓冲区拷贝文件
由于是字符流,因此被拷贝的文件不能是非字符型的文件,例如音频,安装包,视频文件等等。
因为read()方法读取到最后是没有结束值-1的,也不能用readLine()方法,因为结尾可能没有换行符
因此,字符流缓冲区拷贝文件没有比较的意义
总结:
数组依然效率远远高于单字节拷贝,但要考虑内存空间
Buffered缓冲区方式效率比非缓冲区效率高
采用3.4是最快最好的方式