一、 缓冲流
都是在基本的流对象基础之上创建而来的,相当于是对基本流对象的一种增强。
1.1 概述
缓冲流,也叫高效流,是对4个基本的FileXxx
流的增强,所以也是4个流,按照数据类型分类:
- 字节缓冲流:
BufferedInputStream
,BufferedOutputStream
- 字符缓冲流:
BufferedReader
,BufferedWriter
缓冲流的基本原理,是在创建流对象时,会创建一个内置的默认大小的缓冲区数组,通过缓冲区读写,减少系统IO次数,从而提高读写的效率。
1.2 字节缓冲流
(1)BufferedOutputStream:字节缓冲输出流
java.io.BufferedOutputStream extends OutputStream
继承自父类的共性成员方法,OutputStream中的方法也可以使用
构造方法
public BufferedOutputStream(OutputStream out)
: 创建一个新的缓冲输出流,以将数据写入指定的底层输出流public BufferedOutputStream(OutputStream out,int size)
: 创建一个新的缓冲输出流,以将具有指定缓冲区大小的数据写入指定的底层输出流- 参数:
①、OutputStream out:字节输出流;我们可以传递FileOutputStream,缓冲流会给FileOutputStream增加一个缓冲区,提高FileOutputStream的写入效果
②、int size:指定缓冲流内部缓冲区的大小,不指定默认
- 使用步骤(重点):
1.创建FileOutputStream对象,构造方法中绑定要输出的目的地
2.创建BufferedOutputStream对象,构造方法中传递FileOutputStream对象,提高FileOutputStream对象效率
3.使用BufferedOutputStream中的方法write,把数据写入到内部缓冲区中
4.使用BufferedOutputStream对象中的方法flush,把内部缓冲区中的数据,刷新到文件中
5.释放资源(会先调用flush方法刷新数据,第4步可以省略)
public class Demo01BufferedOutputStream{
public static void main(String[] args) throws Exception {
//1.创建FileOutputStream对象,构造方法中绑定要输出的目的地
FileOutputStream fos = new FileOutputStream("D:\\test\\a.txt");
//2.创建BufferedOutputStream对象,构造方法中传递FileOutputStream对象,提高FileOutputStream对象效率
BufferedOutputStream bos = new BufferedOutputStream(fos);
//3.使用BufferedOutputStream中的方法write,把数据写入到内部缓冲区中
bos.write("我把数据写入到内部缓冲区中".getBytes());
//4.使用BufferedOutputStream对象中的方法flush,把内部缓冲区中的数据,刷新到文件中
bos.flush();
//5.释放资源(会先调用flush方法刷新数据,第4步可以省略)
bos.close();
}
}
(2)BufferedInputStream:字节缓冲输入流
java.io.BufferedInputStream extends InputStream
继承自父类的共性成员方法,InputStream中的方法也可以使用
构造方法:
-
public BufferedInputStream(InputStream in)
: 创建一个BufferedInputStream并保存其参数,即输入流in,以便将来使用 -
public BufferedInputStream(InputStream in,int size)
: 创建具有指定缓冲区大侠的BufferedInputStream并保存其参数,即输入流in,以便将来使用 -
参数:
①、InputStream in
:字节输入流;我们可以传递FileInputStream,缓冲流会给FileInputStream增加一个缓冲区,提高FileInputStream in的读取效果
②、int size
:指定缓冲流内部缓冲区的大小,不指定默认 -
使用步骤(重点):
①、创建FileInputStream对象,构造方法中绑定要读取的数据源
②、创建BufferedInputStream对象,构造方法中传递FileInputStream对象,提高FileInputStream对象读取效率
③、使用BufferedInputStream中的方法read,读取文件
④、使用BufferedInputStream
有两种方法都可以读取文件中的数据
public class Demo01BufferedInputStream{
public static void main(String[] args) throws Exception {
//1.创建FileInputStream对象,构造方法中绑定要读取的数据源
FileInputStream fis = new FileInputStream("D:\\test\\a.txt");
//2.创建BufferedInputStream对象,构造方法中传递FileInputStream对象,提高FileInputStream对象读取效率
BufferedInputStream bis = new BufferedInputStream(fis);
//3.使用BufferedInputStream中的方法read,读取文件
//3.1 int read()输入流中读取数据的下一字节
int len = 0;//记录每次读取到的字节
while ((len = fis.read()) !=-1){
System.out.println(len);
}
//4.释放资源
bis.close();
}
}
public class Demo01BufferedInputStream{
public static void main(String[] args) throws Exception {
//1.创建FileInputStream对象,构造方法中绑定要读取的数据源
FileInputStream fis = new FileInputStream("D:\\test\\a.txt");
//2.创建BufferedInputStream对象,构造方法中传递FileInputStream对象,提高FileInputStream对象读取效率
BufferedInputStream bis = new BufferedInputStream(fis);
//3.使用BufferedInputStream中的方法read,读取文件
//3.2 int read(byte[] b) 从输入流中读取一定数量的字节,并将其存储在缓冲区数组b中
byte[] bytes = new byte[1024];//存储每次读取的数据
int len = 0;//读取的有效字节个数
while ((len = bis.read(bytes)) != -1 ){
System.out.println(new String(bytes,0,len));
}
//4.释放资源
bis.close();
}
}
效率测试
查询API,缓冲流读写方法与基本的流是一致的,我们通过复制大文件(375MB),测试它的效率。
- 基本流,代码如下:
public class BufferedDemo {
public static void main(String[] args) throws FileNotFoundException {
// 记录开始时间
long start = System.currentTimeMillis();
// 创建流对象
try (
FileInputStream fis = new FileInputStream("jdk9.exe");
FileOutputStream fos = new FileOutputStream("copy.exe")
){
// 读写数据
int b;
while ((b = fis.read()) != -1) {
fos.write(b);
}
} catch (IOException e) {
e.printStackTrace();
}
// 记录结束时间
long end = System.currentTimeMillis();
System.out.println("普通流复制时间:"+(end - start)+" 毫秒");
}
}
十几分钟过去了...
- 缓冲流,代码如下:
public class BufferedDemo {
public static void main(String[] args) throws FileNotFoundException {
// 记录开始时间
long start = System.currentTimeMillis();
// 创建流对象
try (
BufferedInputStream bis = new BufferedInputStream(new FileInputStream("jdk9.exe"));
BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("copy.exe"));
){
// 读写数据
int b;
while ((b = bis.read()) != -1) {
bos.write(b);
}
} catch (IOException e) {
e.printStackTrace();
}
// 记录结束时间
long end = System.currentTimeMillis();
System.out.println("缓冲流复制时间:"+(end - start)+" 毫秒");
}
}
缓冲流复制时间:8016 毫秒
如何更快呢?
使用数组的方式,代码如下:
public class BufferedDemo {
public static void main(String[] args) throws FileNotFoundException {
// 记录开始时间
long start = System.currentTimeMillis();
// 创建流对象
try (
BufferedInputStream bis = new BufferedInputStream(new FileInputStream("jdk9.exe"));
BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("copy.exe"));
){
// 读写数据
int len;
byte[] bytes = new byte[8*1024];
while ((len = bis.read(bytes)) != -1) {
bos.write(bytes, 0 , len);
}
} catch (IOException e) {
e.printStackTrace();
}
// 记录结束时间
long end = System.currentTimeMillis();
System.out.println("缓冲流使用数组复制时间:"+(end - start)+" 毫秒");
}
}
缓冲流使用数组复制时间:666 毫秒
1.3 字符缓冲流
(1)BufferedWriter:字符缓冲输出流
java.io.BufferedWriter extends Writer
继承自父类的共性成员方法,Writer中的方法也可以使用
构造方法
BufferedWriter(Writer out)
:创建一个使用默认大小输出缓冲区的缓冲字符输出流BufferedWriter(Writer out,int size)
:创建一个使用给定大小输出缓冲区的新缓冲字符输出流- 参数:
①、 Writer out:字符输出流;我们可以传递FileWriter,缓冲流会给FileWriter增加一个缓冲区,提高FileWriter的写入效率
②、int size:指定缓冲区的大小,不写默认大小
特有的成员方法:void newline()
写入一个行分隔符。会根据不同的操作系统获取不同的行分隔符
- 使用步骤:
①、创建字符缓冲输出流对象,构造方法中传递字符输出流
②、调用字符缓冲输出流中的方法write,把数据写入到内存缓冲区中
③、调用字符缓冲输出流中的方法flush,把内存缓冲区中的数据,刷新到文件中
④、释放资源
public class Demo01BufferedWriter{
public static void main(String[] args) throws IOException {
//1、创建字符缓冲输出流对象,构造方法中传递字符输出流
BufferedWriter bw = new BufferedWriter(new FileWriter("D:\\test\\a.txt"));
//2、调用字符缓冲输出流中的方法write,把数据写入到内存缓冲区中
for (int i = 0; i < 10; i++) {
bw.write("海贼王");
//bw.write("\r\n");
//换行
bw.newLine();
}
//3、调用字符缓冲输出流中的方法flush,把内存缓冲区中的数据,刷新到文件中
bw.flush();
//4、释放资源
bw.close();
}
}
(2)BufferedReader:字符缓冲输入流
java.io.BufferedReader extends Reader
继承自父类的共性成员方法,Reader中的方法也可以使用
构造方法
BufferedReader(Reader in)
:创建一个使用默认大小输入缓冲区的缓冲字符输入流BufferedReader(Reader in,int size)
:创建一个使用指定大小输入缓冲区的缓存字符输入流
-
参数:
①、Reader in:字符输入流;我们可以传递FileReader,缓冲流会给FileReader增加一个缓冲区,提高FileReader的读取效率
②、int size:指定缓冲区的大小,不写默认大小 -
特有的成员方法:
String readline():读取一行文本;读取一行数据
行的终止符号,通过下列字符之一即可认为某行已终止:换行(’\n’)、(’\r’)或回车后直接跟着换行(\r\n)
返回值:
包含该行内容的字符串,不包含任何人终止符,如果已到达流末尾,则返回null
- 使用步骤:
①、创建字符缓冲输入流对象,构造方法中传递字符输入流
②、调用字符缓冲输入流中的方法read或者是readline读取文本
③、释放资源
public class Demo01BufferedReader{
public static void main(String[] args) throws IOException {
//1、创建字符缓冲输入流对象,构造方法中传递字符输入流
BufferedReader br = new BufferedReader(new FileReader("D:\\test\\a.txt"));
//2、调用字符缓冲输入流中的方法read或者是readline读取文本
/*
不知道文件中有多少行数据,所以使用while循环
while的结束条件,读取到null结束
*/
String line;
while ((line = br.readLine())!= null){
System.out.println(line);
}
//3、释放资源
br.close();
}
}
运行结果:
海贼王1
海贼王2
海贼王3
1.4 练习:文本排序
请将文本信息恢复顺序。
3.侍中、侍郎郭攸之、费祎、董允等,此皆良实,志虑忠纯,是以先帝简拔以遗陛下。愚以为宫中之事,事无大小,悉以咨之,然后施行,必得裨补阙漏,有所广益。
8.愿陛下托臣以讨贼兴复之效,不效,则治臣之罪,以告先帝之灵。若无兴德之言,则责攸之、祎、允等之慢,以彰其咎;陛下亦宜自谋,以咨诹善道,察纳雅言,深追先帝遗诏,臣不胜受恩感激。
4.将军向宠,性行淑均,晓畅军事,试用之于昔日,先帝称之曰能,是以众议举宠为督。愚以为营中之事,悉以咨之,必能使行阵和睦,优劣得所。
2.宫中府中,俱为一体,陟罚臧否,不宜异同。若有作奸犯科及为忠善者,宜付有司论其刑赏,以昭陛下平明之理,不宜偏私,使内外异法也。
1.先帝创业未半而中道崩殂,今天下三分,益州疲弊,此诚危急存亡之秋也。然侍卫之臣不懈于内,忠志之士忘身于外者,盖追先帝之殊遇,欲报之于陛下也。诚宜开张圣听,以光先帝遗德,恢弘志士之气,不宜妄自菲薄,引喻失义,以塞忠谏之路也。
9.今当远离,临表涕零,不知所言。
6.臣本布衣,躬耕于南阳,苟全性命于乱世,不求闻达于诸侯。先帝不以臣卑鄙,猥自枉屈,三顾臣于草庐之中,咨臣以当世之事,由是感激,遂许先帝以驱驰。后值倾覆,受任于败军之际,奉命于危难之间,尔来二十有一年矣。
7.先帝知臣谨慎,故临崩寄臣以大事也。受命以来,夙夜忧叹,恐付托不效,以伤先帝之明,故五月渡泸,深入不毛。今南方已定,兵甲已足,当奖率三军,北定中原,庶竭驽钝,攘除奸凶,兴复汉室,还于旧都。此臣所以报先帝而忠陛下之职分也。至于斟酌损益,进尽忠言,则攸之、祎、允之任也。
5.亲贤臣,远小人,此先汉所以兴隆也;亲小人,远贤臣,此后汉所以倾颓也。先帝在时,每与臣论此事,未尝不叹息痛恨于桓、灵也。侍中、尚书、长史、参军,此悉贞良死节之臣,愿陛下亲之信之,则汉室之隆,可计日而待也。
案例分析
- 创建HashMap集合对象,可以:存储每行文本的序号(1,2,3…);value:存储每行的文本
- 创建字符缓冲输入流对象,构造方法中绑定字符输入流
- 创建字符缓冲输出流对象,构造方法中绑定字符输出流
- 使用字符缓冲输入流中的方法readline,逐行读取文本
- 对读取到的文本进行切割,获取行中的序号和文本内容
- 把切割好的序号和文本的内容存储到HashMap集合中(key序号,会自动排序)
- 遍历HashMap集合,获取每一个键值对
- 把每一个键值对,拼接为一个文本行
- 把拼接好的文件,使用字符缓冲输出流中的方法write,写入文件中
- 释放资源
案例实现
public class DemoTest{
public static void main(String[] args) throws Exception {
//1.创建HashMap集合对象,可以:存储每行文本的序号(1,2,3.....);value:存储每行的文本
HashMap<String,String> map = new HashMap<>();
//2.创建字符缓冲输入流对象,构造方法中绑定字符输入流
BufferedReader br = new BufferedReader(new FileReader("D:\\test\\a.txt"));
//3.创建字符缓冲输出流对象,构造方法中绑定字符输出流
BufferedWriter bw = new BufferedWriter(new FileWriter("D:\\test\\b.txt"));
//4.使用字符缓冲输入流中的方法readline,逐行读取文本
String line;
while ((line = br.readLine())!=null){
//5.对读取到的文本进行切割,获取行中的序号和文本内容
String[] arr = line.split("\\.");
//6.把切割好的序号和文本的内容存储到HashMap集合中(key序号,会自动排序)
map.put(arr[0],arr[1]);
}
//7.遍历HashMap集合,获取每一个键值对
for (String key:map.keySet()){
String value = map.get(key);
//8.把每一个键值对,拼接为一个文本行
line = key+"."+value;
//9.把拼接好的文件,使用字符缓冲输出流中的方法write,写入文件中
bw.write(line);
bw.newLine();//写换行
}
//10.释放资源
bw.close();
br.close();
}
}