Java IO编程——字节流
前言
想必大家或多或少都有了解过Java IO编程,最开始本人接触Java IO流的时候还挺烦,总是流来流去,理解深度不够。最近重试Java IO编程,想在当下新的理解下,更加深入的以及工程化的学习Java IO,于是本篇博客应运而生!~~~哈哈
Java IO编程中主要分为3个部分:1、File;2、字节流;3、字符流;4、其他流。本博文将介绍较为全面的介绍字节流,并附带实际开发中标准工具类,同时分析以下源码,最终实现从理解到工程开发的目的。 嘀嘀嘀——如果不懂File,可进入本人博客下的传送门☞File入门?
Java IO概述
概念
* IO流用来处理设备之间的数据传输
* Java对数据的操作是通过流的方式
* Java用于操作流的类都在IO包中
* 流按流向分为两种:输入流,输出流。
* 流按操作类型分为两种:
* 字节流 : 字节流可以操作任何数据,因为在计算机中任何数据都是以字节的形式存储的
* 字符流 : 字符流只能操作纯字符数据,比较方便。
IO流常用父类
* 字节流的抽象父类:
* InputStream
* OutputStream
* 字符流的抽象父类:
* Reader
* Writer
字节流入门
在IO字节流中,又分为InputStream(输入流)、OutputStream(输出流)。我个人对输入输出流的概念总是弄混淆,这里给大家介绍一下我白般琢磨出的记忆方法:
输入流:我们从流的角度来想,输入流就是将文件中的内容输入到流中,及读(read)文件。
输出流:还是从流的角度来看,输出流就是将文件从流中输出到文件中,及写(write)文件。
FileInputStream&FileOutputStream
FileInputStream:实现了InputStream接口,主要通过read()方法来读文件,read很多重载方法可以实现一次读取多个或特定位数的方法。
read()一次读取一个字节返回int
FileInputStream fis = new FileInputStream("aaa.txt"); //创建一个文件输入流对象,并关联aaa.txt
int b; //定义变量,记录每次读到的字节
while((b = fis.read()) != -1) { //将每次读到的字节赋值给b并判断是否是-1
System.out.println(b); //打印每一个字节
}
fis.close();
FileOutputStream:实现了OutputStream接口,主要通过write()方法写文件,write很多重载方法可以实现一次读取多个或特定位数的方法。write()一次写出一个字节
FileOutputStream fos = new FileOutputStream("bbb.txt"); //如果没有bbb.txt,会创建出一个
//fos.write(97); //虽然写出的是一个int数,但是在写出的时候会将前面的24个0去掉,所以写出的是一个byte
fos.write(98);
fos.write(99);
fos.close();
问:read()方法读取的是一个字节,为什么返回是int,而不是byte?
因为字节输入流可以操作任意类型的文件,比如图片音频等,这些文件底层都是以二进制形式的存储的,如果每次读取都返回byte,有可能在读到中间的时候遇到111111111那么这11111111是byte类型的-1,我们的程序是遇到-1就会停止不读了,后面的数据就读不到了,所以在读取的时候用int类型接收,如果11111111会在其前面补上24个0凑足4个字节,那么byte类型的-1就变成int类型的255了这样可以保证整个数据读完,而结束标记的-1就是int类型。
FileOutputStream追加
FileOutputStream的构造方法写出数据实现数据的追加写入
FileOutputStream fos = new FileOutputStream("bbb.txt",true); //如果没有bbb.txt,会创建出一个
//fos.write(97); //虽然写出的是一个int数,但是在写出的时候会将前面的24个0去掉,所以写出的一个byte
fos.write(98);
fos.write(99);
fos.close();注:FileOutputStream普通构造方式FileInputStream fis = new FileInputStream("aaa.txt");是先检擦aaa.txt文件是否存在,如果不存在创建,如果存在则清空文件,然后写入像写入的内容,这也就导致无法在原来写入内容的基础上接着写入,所以才需要引入FileOutputStream的构造方法写出数据实现数据的追加写入。
字节流拷贝文件的四种方式
方式一:单字节拷贝,如下图一,缺点:效率低。
方式二:一次性整体拷贝,如下图二,缺点:当拷贝文件过大时,容易造成内存溢出。
* int read(byte[] b):一次读取一个字节数组
* write(byte[] b):一次写出一个字节数组
* available()获取读的文件所有的字节个数方式三:小数组分批拷贝,如下图三。(推荐)
方式四:BufferedInputStream&BufferedOutputStream包装拷贝,如下图四。(推荐)
缓冲思想
* 字节流一次读写一个数组的速度明显比一次读写一个字节的速度快很多,
* 这是加入了数组这样的缓冲区效果,java本身在设计的时候,
* 也考虑到了这样的设计思想(装饰设计模式后面讲解),所以提供了字节缓冲区流
BufferedInputStream
* BufferedInputStream内置了一个缓冲区(数组)
* 从BufferedInputStream中读取一个字节时
* BufferedInputStream会一次性从文件中读取8192个, 存在缓冲区中, 返回给程序一个
* 程序再次读取时, 就不用找文件了, 直接从缓冲区中获取
* 直到缓冲区中所有的都被使用过, 才重新从文件中读取8192个
BufferedOutputStream
* BufferedOutputStream也内置了一个缓冲区(数组)
* 程序向流中写出字节时, 不会直接写到文件, 先写到缓冲区中
* 直到缓冲区写满, BufferedOutputStream才会把缓冲区中的数据一次性写到文件里
问:小数组的读写和带Buffered的读取哪个更快?
定义小数组如果是8192个字节大小和Buffered比较的话,定义小数组会略胜一筹,因为读和写操作的是同一个数组而Buffered操作的是两个数组,故定义小数组略微快点,但是在实际工程中两个都是推荐使用的。
close()&flush()方法区分
flush()方法
* 用来刷新缓冲区的,刷新后可以再次写出
close()方法
* 用来关闭流释放资源的的,如果是带缓冲区的流对象的close()方法,不但会关闭流,还会再关闭流之前刷新缓冲区,关闭后不能再写出
IO流(字节流读写中文)
字节流读取中文的问题
* 字节流在读中文的时候有可能会读到半个中文,造成乱码 .如下图:
字节流写出中文的问题(如下图二)
* 字节流直接操作的字节,所以写出中文必须将字符串转换成字节数组
* 写出回车换行 write("\r\n".getBytes());
IO流标准
流的标准处理异常代码1.6版本及其以前,如下图:
流的标准处理异常代码1.7版本
* 原理:在try()中创建的流对象必须实现了AutoCloseable这个接口,如果实现了,在try后面的{}(读写代码)执行后就会自动调用,流对象的close方法将流关掉;FileInputStream和FileOutputStream类底层都实现了AutoCloseable中的close方法如下图:
JDK1.7标准IO处理实现如下:
实战练习
题目一:简单图片加密
部分代码实现如下:
//图片加密
BufferedInputStream bis = new BufferedInputStream(new FileInputStream("a.jpg"));
BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("b.jpg"));
int b;
while((b = bis.read()) != -1) {
bos.write(b ^ 123); //write的时候进行异或运算,第二次异或运算就可以还原得到b
}
bis.close();
bos.close();
题目二:录入数据拷贝到文件)
将键盘录入的数据拷贝到当前项目下的text.txt文件中,键盘录入数据当遇到quit时就退出
部分代码实现如下:
//写入文件部分代码实现
Scanner sc = new Scanner(System.in);
FileOutputStream fos = new FileOutputStream("text.txt");
System.out.println("请输入:");
while(true) {
String line = sc.nextLine();
if("quit".equals(line))
break;
fos.write(line.getBytes());
fos.write("\r\n".getBytes());
}
fos.close();
IO拷贝纯文本工具类下载:GetHub(免费) 、 CSDN文档(付费)。
谢谢阅读 ----知飞翀