前面介绍了利用文件写入器和文件读取器来读写文件,因为FileWriter与FileReader读写的数据以字符为单位,所以这种读写文件的方式被称作“字符流I/O”,其中字母I代表输入Input,字母O代表输出Output。可是FileWriter的读操作并不高效,缘由在于FileWriter每次调用write方法都会直接写入文件,假如某项业务需要多次调用write方法,那么程序就会写入文件同样次数。因为写文件本质是写磁盘,磁盘的速度远不如内存,所以频繁地写文件必然严重降低程序的运行效率。为此Java又设计了缓存写入器BufferedWriter,它的write方法并不直接写入文件,而是先写入一块缓存,等到缓存写满了再将缓存上的数据写入文件。由于缓存空间位于内存之中,写入缓存等同访问内存,这样相当于把写磁盘动作替换成写内存动作,因此BufferedWriter的整体写文件性能要大大优于FileWriter。除此之外,BufferedWriter还新增了下列几个方法:
newLine:往文件末尾添加换行标记(Window系统是回车加换行)。当然实际上是先往缓存添加换行标记,并非直接往磁盘写入换行标记。
flush:立即将缓冲区中的数据写入磁盘。默认情况要等缓冲区满了才会写入磁盘,或者调用close方法关闭文件之时也会写入磁盘,但是有时程序猴急,一定要立即写入磁盘,此时就需调用flush方法强行写磁盘。
使用缓存写入器之前要先创建文件读取器对象,并获得父类Writer的实例,然后再据此创建缓存写入器对象。下面是通过缓存写入器把多行字符串写入文件的代码例子:

	private static String mSrcName = "D:/test/aad.txt";
// 使用缓存字符流写入文件
private static void writeBuffer() {
String str1 = "白日依山尽,黄河入海流。";
String str2 = "欲穷千里目,更上一层楼。";
File file = new File(mSrcName); // 创建一个指定路径的文件对象
// try(...)允许在圆括号内部拥有多个资源创建语句,语句之间以冒号分隔
// 先创建文件写入器,再根据文件读取器创建缓存写入器
try (Writer writer = new FileWriter(file);
BufferedWriter bwriter = new BufferedWriter(writer);) {
// FileWriter的每次write调用都会直接写入磁盘,不但效率低,性能也差。
// BufferedWriter的每次write调用会先写入缓冲区,直到缓冲区满了才写入磁盘,
// 缓冲区大小默认是8K,查看源码defaultCharBufferSize = 8192;
// 资源释放的close方法再把缓冲区的剩余数据写入磁盘,
// 或者中途调用flush方法也可提前将缓冲区的数据写入磁盘。
bwriter.write(str1); // 往文件写入字符串
bwriter.newLine(); // 另起一行,也就是在文件末尾添加换行标记(Window系统是回车加换行)
bwriter.write(str2); // 往文件写入字符串
//bwriter.flush(); // 把缓冲区中的数据写入磁盘
} catch (Exception e) {
e.printStackTrace();
}
}

既然文件写入器有对应的缓存写入器,那么文件读取器也有对应的缓存读取器BufferedReader。BufferedReader的实现原理与它的兄弟BufferedWriter类似,另外BufferedReader比起文件读取器新增了如下方法:
readLine:从文件中读取一行数据。
mark:在当前位置做个标记。
reset:重置文件指针,令其回到上次标记的位置。也就是回到上次mark方法标记的文件位置。
lines:读取文件内容的所有行,返回的是Stream<String>流对象,之后便可按照流式处理来加工该字符串流。
若想使用缓存读取器,依然要先创建文件读取器,再根据其父类的读取器实例创建缓存读取器。下面是通过缓存读取器从文件中读取多行字符串的代码例子:

	// 使用缓存字符流读取文件
private static void readBuffer() {
File file = new File(mSrcName); // 创建一个指定路径的文件对象
// try(...)允许在圆括号内部拥有多个资源创建语句,语句之间以冒号分隔
// 先创建文件读取器,再根据文件读取器创建缓存读取器
try (Reader reader = new FileReader(file);
BufferedReader breader = new BufferedReader(reader);) {
breader.mark((int) file.length()); // 做个标记
for (int i=1; ; i++) {
// FileReader只能一个字符一个字符地读,或者一次性读进字符数组。
// BufferedReader还支持一行一行地读。
String line = breader.readLine(); // 从文件中读出一行文字
if (line == null) { // 读到了空指针,表示已经到了文件末尾
break;
}
System.out.println("第"+i+"行的文字为:"+line);
}
breader.reset(); // 重置文件指针,令其回到上次标记的位置
for (int i=1; ; i++) {
String line = breader.readLine(); // 从文件中读出一行文字
if (line == null) { // 读到了空指针,表示已经到了文件末尾
break;
}
System.out.println("又读了一遍 第"+i+"行的文字为:"+line);
}
//breader.lines(); // 返回Stream<String>对象,之后可按照流式处理来加工该字符串流
} catch (Exception e) {
e.printStackTrace();
}
}

注意到以上代码BufferedWriter和BufferedReader的创建语句都位于try后面的圆括号之中,这是因为Writer与Reader两大家族统统实现了AutoCloseable接口,所以由它们繁衍而来的所有子类都具备自动释放资源的功能。另外,try语句支持同时管理多个资源类,只要它们的对象创建语句以冒号隔开,程序在运行时即可自动回收相关的资源。
结合运用读操作和写操作,可以实现文件复制的功能,无非是一边从源文件中读出数据,另一边紧接着往目标文件写入数据。采用缓存读取器和缓存写入器逐行复制的话,具体的文件复制代码示例如下:

	private static String mSrcName = "D:/test/aad.txt";
private static String mDestName = "D:/test/aad_copy.txt";
// 通过缓存字符流逐行复制文件
private static void copyFile() {
File src = new File(mSrcName); // 创建一个指定路径的源文件对象
File dest = new File(mDestName); // 创建一个指定路径的目标文件对象
// try(...)允许在圆括号内部拥有多个资源创建语句,语句之间以冒号分隔
// 分别创建源文件的缓存读取器,以及目标文件的缓存写入器
try (BufferedReader breader = new BufferedReader(new FileReader(src));
BufferedWriter bwriter = new BufferedWriter(new FileWriter(dest));) {
for (int i=0; ; i++) {
String line = breader.readLine(); // 从文件中读出一行文字
if (line == null) { // 读到了空指针,表示已经到了文件末尾
break;
}
if (i != 0) { // 第一行开头不用换行
bwriter.newLine(); // 另起一行,也就是在文件末尾添加换行标记
}
bwriter.write(line); // 往文件写入字符串
}
} catch (Exception e) {
e.printStackTrace();
}
System.out.println("文件复制完成,源文件大小="+src.length()+",新文件大小="+dest.length());
}

或者也可逐个字符来复制文件,此时BufferedReader每次调用的read方法只返回整型数表示一个字符,并且BufferedWriter每次调用的write方法也只写入该字符对应的整型数。通过依次遍历源文件的所有字符,同时往目标文件依次写入这些字符,从而完成逐个字符复制文件的操作流程。下面是采取逐字符复制文件的代码例子:

	// 通过缓存字符流逐个字符复制文件
private static void copyFileByInt() {
File src = new File(mSrcName); // 创建一个指定路径的源文件对象
File dest = new File(mDestName); // 创建一个指定路径的目标文件对象
// try(...)允许在圆括号内部拥有多个资源创建语句,语句之间以冒号分隔
// 分别创建源文件的缓存读取器,以及目标文件的缓存写入器
try (BufferedReader breader = new BufferedReader(new FileReader(src));
BufferedWriter bwriter = new BufferedWriter(new FileWriter(dest));) {
while (true) { // 开始遍历文件中的所有字符
int temp = breader.read(); // 从源文件中读出一个字符
if (temp == -1) { // read方法返回-1表示已经读到了文件末尾
break;
}
bwriter.write(temp); // 往目标文件写入一个字符
}
} catch (Exception e) {
e.printStackTrace();
}
System.out.println("文件复制完成,源文件大小="+src.length()+",新文件大小="+dest.length());
}

需要注意的是,使用字符流复制文件只有逐行复制和逐字符复制两种方式,不可采取整个读到字符数组再整个写入字符数组的方式。之所以不能通过字符数组复制文件,是因为中文跟英文不一样,一个汉字会占用多个字节(GBK编码的每个汉字占用两个字节,UTF8编码的每个汉字占用三个字节)。若要把文件内容读到字符数组,势必先得知晓该数组的长度,可是调用文件对象的length方法只能得到该文件的字节长度,并非字符长度。譬如“白日依山尽”这个字符串在内存中的字符数组长度为5,写到UTF8编码的文件之后,文件大小是5*3=15字节;接着想把文件内容读到字符数组,然而15字节的文件天晓得它有几个字符,可能有5个UTF8编码的中文字符,也可能有15个英文字符,也可能有5个GBK编码的中文字符加5个英文字符共10个字符,总之你根本想不到该分配多大的字符数组。既然确定不了待读取的字符数组长度,就无法一字不差地复制文件内容了。

更多Java技术文章参见《Java开发笔记(序)章节目录

Java开发笔记(八十六)通过缓冲区读写文件的更多相关文章

  1. Java开发笔记(十六)非此即彼的条件分支

    前面花了大量篇幅介绍布尔类型及相应的关系运算和逻辑运算,那可不仅仅是为了求真值或假值,更是为了通过布尔值控制流程的走向.在现实生活中,常常需要在岔路口抉择走去何方,往南还是往北,向东还是向西?在Jav ...

  2. Java开发笔记(九十六)线程的基本用法

    每启动一个程序,操作系统的内存中通常会驻留该程序的一个进程,进程包含了程序的完整代码逻辑.一旦程序退出,进程也就随之结束:反之,一旦强行结束进程,程序也会跟着退出.普通的程序代码是从上往下执行的,遇到 ...

  3. Java开发学习(三十六)----SpringBoot三种配置文件解析

    一. 配置文件格式 我们现在启动服务器默认的端口号是 8080,访问路径可以书写为 http://localhost:8080/books/1 在线上环境我们还是希望将端口号改为 80,这样在访问的时 ...

  4. Java开发学习(二十六)----SpringMVC返回响应结果

    SpringMVC接收到请求和数据后,进行了一些处理,当然这个处理可以是转发给Service,Service层再调用Dao层完成的,不管怎样,处理完以后,都需要将结果告知给用户. 比如:根据用户ID查 ...

  5. Java开发笔记(十八)上下求索的while循环

    循环是流程控制的又一重要结构,“白天-黑夜-白天-黑夜”属于时间上的循环,古人“年复一年.日复一日”的“日出而作.日落而息”便是每天周而复始的生活.计算机程序处理循环结构时,给定一段每次都要执行的代码 ...

  6. Java学习笔记(十六)——Java RMI

    [前面的话] 最近过的好舒服,每天过的感觉很充实,一些生活和工作的技巧注意了就会发现,其实生活也是可以过的如此的有滋有味,满足现在的状况,并且感觉很幸福. 学习java RMI的原因是最近在使用dub ...

  7. Java开发笔记(十九)规律变化的for循环

    前面介绍while循环时,有个名叫year的整型变量频繁出现,并且它是控制循环进出的关键要素.不管哪一种while写法,都存在三处与year有关的操作,分别是“year = 0”.“year<l ...

  8. .net开发笔记(十六) 对前部分文章的一些补充和总结

    补充有两个: 一个是系列(五)中讲到的事件编程(网址链接),该文提及到了事件编程的几种方式以及容易引起的一些异常,本文补充“多线程事件编程”这一块. 第二个是前三篇博客中提及到的“泵”结构在编程中的应 ...

  9. 【Java学习笔记之十六】浅谈Java中的继承与多态

    1.  什么是继承,继承的特点? 子类继承父类的特征和行为,使得子类具有父类的各种属性和方法.或子类从父类继承方法,使得子类具有父类相同的行为. 特点:在继承关系中,父类更通用.子类更具体.父类具有更 ...

  10. Java开发笔记(十)一元运算符的技巧

    前面讲到赋值运算符的时候,提到“x = x+7”可以被“x += 7”所取代,当然Java编程中给某个变量自加7并不常见,常见的是给某变量自加1,就像走台阶,一般都是一级一级台阶地走,犯不着一下子跳上 ...

随机推荐

  1. Hadoop3.0 WordCount测试一直Accept 状态,Nodes of the cluster 页面node列表个数为0

    起因是我运行wordcount测试一直卡主,不能执行,一直处于 Accept 状态,等待被执行,刚开始是各种配置yarn参数,以及host配置,后来发现还是不行 hadoop 集群安装完成后,在500 ...

  2. 我的Python之旅第四天

    一 名称空间.作用域.取值顺序 1 名称空间 当程序运行时,代码从上至下依次执行,它会将变量与值得关系存储在一个空间中,这个空间就叫做名称空间,也叫命名空间.全局名称空间. 当程序遇到函数时,他会将函 ...

  3. 从壹开始前后端分离[.NetCore ] 38 ║自动初始化数据库(不定期更新)

    缘起 哈喽大家好呀,我们又见面啦,这里先祝大家圣诞节快乐哟,昨天的红包不知道有没有小伙伴抢到呢.今天的这篇内容灰常简单,只是对我们的系统的数据库进行CodeFirst,然后就是数据处理,因为这几个月来 ...

  4. asp.net core系列 42 Web 应用 分部视图

    一.分部视图 对于MVC 视图和 Razor Pages 页面,都有分部视图功能.通常将 MVC 视图和 Razor Pages 页面统称为“标记文件”,下面会常提到该名词.使用分部视图的优势包括:( ...

  5. 深入理解pandas读取excel,txt,csv文件等命令

    pandas读取文件官方提供的文档 在使用pandas读取文件之前,必备的内容,必然属于官方文档,官方文档查阅地址 http://pandas.pydata.org/pandas-docs/versi ...

  6. Python爬虫入门教程 59-100 python爬虫高级技术之验证码篇5-极验证识别技术之二

    图片比对 昨天的博客已经将图片存储到了本地,今天要做的第一件事情,就是需要在两张图片中进行比对,将图片缺口定位出来 缺口图片 完整图片 计算缺口坐标 对比两张图片的所有RBG像素点,得到不一样像素点的 ...

  7. JavaScript构造函数

    JavaScript不同于其他强类型语言,没有类的概念,但是它支持可以与实例共同使用特殊的Constructor构造器,使用new关键字创建新的实例,并告知JavaScript使用对象的内规则去定制这 ...

  8. 你需要知道的这几种 asp.net core 修改默认端口的方式

    一般情况下,aspnetcore发布后的默认端口是5000,这个大家都知道,而且默认骨架代码中没有看到任何让你输入的ip地址和端口号,但作为程序员的我们,不希望 被框架所管制,那如何实现默认端口的修改 ...

  9. vue安装jquery和配置(不需要在页面引入直接可以使用)

    首先在命令行工具上输入 npm install jquery --save-dev 安装完成之后在build文件夹下的webpack.base.conf.js进行配置,在顶部添加:const webp ...

  10. 通过免费开源ERP构建业界领先的供应链+垂直电商平台成功案例分享

    案例客户简介 Healey Green是一家新成立的企业,在线销售和销售园艺机械. 他们的产品范围包括草坪割草机,割灌机,地钻,链锯等. 在一个竞争非常激烈的市场中,这位雄心勃勃的新人将开始接受那些以 ...