文件输出流FileOutputStream跟FileWriter同样有个毛病,每次调用write方法都会直接写到磁盘,使得频繁的写操作性能极其低下。正如FileWriter搭上了缓存兄弟BufferedWriter那样,FileOutputStream也有自己的缓存兄弟BufferedOutputStream,这个缓存输出流的用法与缓存写入器非常相似,主要体现在如下三点:
1、每次创建缓存输出流对象之前,都要先构建文件输出流对象,然后据此构建缓存输出流对象;
2、它的write方法先把数据写到缓存,等到缓存满了才写入磁盘,或者在调用close方法关闭文件之时将缓存数据写入磁盘。
3、缓存输出流仍然提供了flush方法,调用flush方法表示立即把缓存中的数据写入磁盘。
下面是利用缓存输出流写文件的代码例子:

	private static String mSrcName = "D:/test/aaf.txt";
// 利用缓存输出流写入文件
private static void writeBuffer() {
String str = "白日依山尽,黄河入海流。\n欲穷千里目,更上一层楼。";
// 根据指定文件路径构建文件输出流对象,然后据此构建缓存输出流对象
try (FileOutputStream fos = new FileOutputStream(mSrcName);
BufferedOutputStream bos = new BufferedOutputStream(fos)) {
bos.write(str.getBytes()); // 把字节数组写入缓存输出流
//bos.flush(); // 立即写入磁盘。如果不立即写入,最后调用close方法时也会写入
} catch (Exception e) {
e.printStackTrace();
}
}

看过了缓存输出流,再来看缓存输入流BufferedInputStream。由于字节流操作的数据形式为字节数组,因此不管是缓存输出流还是缓存输出流,都不提供按行读写的功能。若想创建缓存输入流对象,依旧要先构建文件输入流对象,再据此构建缓存输入流对象。另外BufferedInputStream保留了mark和reset两个方法,前者用于在当前位置做个标记,后者可重置输入流指针,令其回到上次标记的位置。下面是利用缓存输入流读文件的代码例子:

	// 利用缓存输入流读取文件
private static void readBuffer() {
// 根据指定文件路径构建文件输入流对象,然后据此构建缓存输入流对象
try (FileInputStream fis = new FileInputStream(mSrcName);
BufferedInputStream bis = new BufferedInputStream(fis)) {
// 分配长度为文件大小的字节数组。available方法返回当前位置后面的剩余部分大小
byte[] bytes = new byte[bis.available()];
bis.read(bytes); // 从缓存输入流中读取字节数组
// 缓存输入流的mark和reset用法类似于BufferedReader的同名方法
//bis.mark(bis.available()); // 在当前位置做个标记
//bis.reset(); // 重置输入流指针,令其回到上次标记的位置
String content = new String(bytes); // 把字节数组转换为字符串
System.out.println("content="+content);
} catch (Exception e) {
e.printStackTrace();
}
}

因为字节流只处理字节数组,不处理字符数组,所以借助于字节数组可以很轻松地在输入和输出流之间转换。调用缓存输入流对象的read方法,将文件数据读到指定的字节数组;然后调用缓存输出流对象的write方法,马上把刚读取的字节数组写入文件,一进一出之间就顺带完成了文件复制功能。下面是通过缓存输入和输出流复制文件的代码例子:

	private static String mSrcName = "D:/test/aaf.txt";
private static String mDestName = "D:/test/aaf_copy.txt";
// 利用缓存输入和输出流复制文件
private static void copyFile() {
// 分别构建缓存输入流对象和缓存输出流对象
try (BufferedInputStream bis = new BufferedInputStream(new FileInputStream(mSrcName));
BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream(mDestName))) {
// 分配长度为文件大小的字节数组。available方法返回当前位置后面的剩余部分大小
byte[] bytes = new byte[bis.available()];
bis.read(bytes); // 从缓存输入流中读取字节数组
bos.write(bytes); // 把字节数组写入缓存输出流
System.out.println("文件复制完成,源文件大小="+bytes.length+",新文件大小="+bytes.length);
} catch (Exception e) {
e.printStackTrace();
}
}

同之前介绍的通过缓存读取器和写入器复制文件相比,缓存输入和输出流在复制文件的时候去掉了恼人的循环语句,整个实现代码显得更加精练和高效。

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

Java开发笔记(八十九)缓存字节I/O流的更多相关文章

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

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

  2. Java开发笔记(九十九)定时器与定时任务

    前面介绍了线程的几种运行方式,不管哪种方式,一旦调用了线程实例的start方法,都会立即启动线程的事务处理.然而某些业务场景在事务执行时间方面有特殊需求,例如期望延迟若干时间之后才开始事务运行,又如期 ...

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

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

  4. .Net开发笔记(十九) 创建一个可以可视化设计的对象

    阅读本篇博客之前需要了解VS窗体设计器的工作原理,详细可参见本系列博客(十).(十一).(十二).必须需要知道的一条结论就是:处于窗体设计器(Form Designer)中的任何组件(包含控件,下同) ...

  5. Java学习笔记(十九)——Java 日志记录 AND log4j

    [前面的话] 学习的进度应该稍微在快一点. Java日志到了必须学习怎么使用的时候了,因为在项目中要进行使用.基础性文章,选择性阅读. [结构] java日志对调试,记录运行,问题定位都起到了很重要的 ...

  6. 【Java学习笔记之十九】super在Java继承中的用法小结

    1)有人写了个很好的初始化属性的构造函数,而你仅仅想要在其中添加另一些自己新建属性的初始化,这样在一个构造函数中调用另外一个构造函数,可以避免重复的代码量,减少工作量: 2)在一个构造函数中调用另外一 ...

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

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

  8. Java开发笔记(十二)布尔变量论道与或非

    在编程语言的设计之初,它们除了可以进行数学计算,还常常用于逻辑推理和条件判断.为了实现逻辑判断的功能,Java引入了一种布尔类型boolean,用来表示“真”和“假”.该类型的变量只允许两个取值,即t ...

  9. Java开发笔记(十四)几种运算符的优先级顺序

    到目前为止,我们已经学习了Java语言的好几种运算符,包括算术运算符.赋值运算符.逻辑运算符.关系运算符等基础运算符,并且在书写赋值语句时都没添加圆括号,显然是默认了先完成算术.逻辑.关系等运算,最后 ...

  10. Java开发笔记(十五)短路逻辑运算的优势

    前面提到逻辑运算只能操作布尔变量,这其实是不严谨的,因为经过Java编程实现,会发现“&”.“|”.“^”这几个逻辑符号竟然可以对数字进行运算.譬如下面的代码就直接对数字分别开展了“与”.“或 ...

随机推荐

  1. Tornado day1

    Tornado 之路由配置 首先导入模块,使用Application方法中可配置多个路由,格式必须为列表中是元组 元组的第一个是配置的url,第二个参数时自定义的类(继承自RequestHandler ...

  2. Python_回调函数

    import os import stat def remove_readonly(func,path): #定义回调函数 os.chmod(path,stat.S_IWRITE) #删除文件的只读文 ...

  3. Scrapy爬虫框架补充内容三(代理及其基本原理介绍)

    前言:(本文参考维基百科及百度百科所写) 当我们使用爬虫抓取数据时,有时会产生错误比如:突然跳出来了403 Forbidden 或者网页上出现以下提示:您的ip访问频率太高 或者时不时跳出一个验证码需 ...

  4. 一个能拖动,能调整大小,能更新bind值的vue指令-vuedragx

    一. 背景说明 开发一个可自定义组件化门户配置页面,期间采用了vue框架作为前端视图引擎,作为一个刚入手vue的萌新,开发第一个功能就遇到了拦路虎.需要一个拖动并且可改变大小的容器盒子.当时查看vue ...

  5. Eclipse插件:mybatis generator的使用步骤

    一.首先,安装eclipse插件 Help--Eclipser Marketplace中查找:Mybatis Generator 1.3.5安装 二.新建project New--other--查找如 ...

  6. 深入NGINX:nginx高性能的实现原理

    深入NGINX:我们如何设计它的性能和扩展性 来源: cnBeta  原文链接 英文原文:Inside NGINX: How We Designed for Performance & Sca ...

  7. Java 面试知识点解析(五)——网络协议篇

    前言: 在遨游了一番 Java Web 的世界之后,发现了自己的一些缺失,所以就着一篇深度好文:知名互联网公司校招 Java 开发岗面试知识点解析 ,来好好的对 Java 知识点进行复习和学习一番,大 ...

  8. linux,windows下检测指定的IP地址是否可用或者检测IP地址冲突的3种方式(批处理程序,python程序,linux shell 批量ping)

    本文中的脚本适用范围: 1)检测某些IP地址是否被占用: 2)检测网络中某些设备是否存活: 3)在分配新的ip地址之前,批量检测环境中是否存在冲突的机器 以上检测基于ICMP Ping报文,要求所有的 ...

  9. MIT KIT OpenID Connect Demo Client

    Hello world! You are NOT currently logged in. This example application is configured with several pa ...

  10. 解决持久化数据太大,单个节点的硬盘无法存储的问题;解决运算量太大,单个节点的内存、CPU无法处理的问题

    需要学习的技术很多,要自学新知识也不是一件容易的事,选择一个自己比较感兴趣的会是一个比较好的开端,于是,打算学一学分布式系统. 带着问题,有目的的学习,先了解整体架构,在深入感兴趣的细节,这是我的计划 ...