Java中IO流讲解(一)
一、概念
- IO流用来处理设备之间的数据传输
- Java对数据的操作是通过流的方式
- Java用于操作流的类都在IO包中
- 流按流向分为两种:输入流,输出流
- 流按操作类型分为两种:
- 字节流 : 字节流可以操作任何数据,因为在计算机中任何数据都是以字节的形式存储的
- 字符流 : 字符流只能操作纯字符数据,比较方便。
二、字节流
字节流的抽象父类:InputStream和OutputStream,由于抽象类是不能直接new对象的,所以我们可以使用它们的子类来创建对象,下面我们主要来学习对文件的操作。FileInputStream对应文件输入流,FileOutputStream对应文件输出流。
1、FileInputStream常用方法介绍:
构造方法 |
FileInputStream(String name) 传入的参数为一个有效的文件路径,创建文件输入流对象 FileInputStream(File file) 传入的参数为一个文件对象,创建文件 |
成员方法 |
public int available() throws IOException 返回剩余可读的字节数 public void close() throws IOException 关闭流对象 public int read() throws IOException 读取一个字节,返回值为读取到的字节的int类型值,读取到文件末尾返回-1 public int read(byte[] b) throws IOException 最多读取b.length 个字节到字节数组b,返回读取到的字节数,读取到文件末尾返回-1 |
2、从文件上读取一个字节 read()
a.txt内容如下:
ab
import java.io.FileInputStream; public class FileInputStreamTest { public static void main(String[] args) throws Exception { FileInputStream fis = new FileInputStream("a.txt"); int x = fis.read(); //从硬盘上读取一个字节 System.out.println("x = " + x); int y = fis.read(); System.out.println("y = " + y); int z = fis.read(); System.out.println("z = " + z); fis.close(); //关闭流 } } 输出: x = 97 y = 98 z = -1
read()方法一次读取一个字节,并把读取到的字节赋值成int类型,a对应的ascii码为97,b对应的为98,当读到文件末尾的时候,方法返回-1。有了上面的代码学习,我们可以对代码进行改进一下:
import java.io.FileInputStream; public class FileInputStreamTest { public static void main(String[] args) throws Exception { FileInputStream fis = new FileInputStream("a.txt"); int a; while((a = fis.read()) != -1) { System.out.println(a); } fis.close(); } } 输出: 97 98
3、文件内容读取到字节数组 read(byte[] b)
a.txt内容如下:
abc
import java.io.FileInputStream; public class FileInputStreamTest { public static void main(String[] args) throws Exception { FileInputStream fis = new FileInputStream("a.txt"); byte[] arr = new byte[2]; //定义长度2的字节数组 int a = fis.read(arr); //一次最多将arr.length个字节读取到arr中,返回读取到的字节数 System.out.println("读取到的字节个数:" + a); for (byte b : arr) { System.out.println(b); } System.out.println("-----------"); a = fis.read(arr); System.out.println("读取到的字节个数:" + a); for (byte b : arr) { System.out.println(b); } fis.close(); } } 输出:读取到的字节个数:2 97 98 ----------- 读取到的字节个数:1 99 98
第一次读取的时候,把a和b对应的97和98读取到arr,读取到的字节数是2,这个很好理解。看第二次的输出,第二次读取到1个字节也很好理解,因为a.txt的内容是abc,第一次ab都读过了,第二次就只读了剩下的c,但是字节数组输出是99和98。99对应的是c,但是怎么还会有个98(对应的是b)呢?这是因为在第一次读取的时候,97和98分别保存到了字节数组的0和1号索引。第二次读取到c的时候,用99覆盖了0号索引的97,而98还在1号索引。所以输出99和98
4、FileOutputStream常用方法介绍:
构造方法 |
FileOutputStream(File file) 创建向指定 File 对象表示的文件中写入数据的文件输出流 FileOutputStream(File file, boolean append) 和上面的方法的区别是,参数append如果是true表示在文件末尾追加。否则就覆盖 FileOutputStream(String name) 创建一个向具有指定名称的文件中写入数据的输出文件流。 FileOutputStream(String name, boolean append) 参数append的意思和FileOutputStream(File file, boolean append)一致 |
成员方法 |
public void close() throws IOException 关闭流对象 public void write(int b) throws IOException 将指定字节写入此文件输出流。 public void write(byte[] b) throws IOException 将 b.length 个字节从指定 byte 数组写入此文件输出流中 public void write(byte[] b, int off, int len) throws IOException 将指定 byte 数组中从偏移量 off 开始的 len 个字节写入此文件输出流。 |
import java.io.FileOutputStream; public class FileOutputStreamTest { public static void main(String[] args) throws Exception { // 如果没有就自动创建这个文件 FileOutputStream fos = new FileOutputStream("b.txt"); // 虽然写出的是int类型,但是写到文件的时候会自动去除前三个8位 fos.write(98); fos.close(); } }
文件b.txt的内容为:b
6、向文件写出字节数组的内容 write(byte[] b)
import java.io.FileOutputStream; public class FileOutputStreamTest { public static void main(String[] args) throws Exception { FileOutputStream fos = new FileOutputStream("b.txt"); byte[] arr = "hello,world!".getBytes(); fos.write(arr); fos.close(); } }
文件b.txt的内容为:hello,world!
三、文件内容拷贝
有了上面的学习后,我们就可以拷贝文件了。下面来看个例子,要求把a.txt的内容拷贝到b.txt
import java.io.FileInputStream; import java.io.FileOutputStream; public class FileCopyTest { public static void main(String[] args) throws Exception { FileInputStream fis = new FileInputStream("a.txt"); FileOutputStream fos = new FileOutputStream("b.txt"); byte[] arr = new byte[1024 * 8]; //定义字节数组的长度为8K int len; while((len = fis.read(arr)) != -1) { //len保存的是每次读取的字节数,保存到arr中 fos.write(arr, 0, len); //把arr的len个字节写出文件,0代表开始字节的索引位置 } fis.close(); fos.close(); } }
上面的这几行代码就是文件拷贝的核心代码,所有的其他文件拷贝方式都是在这几行代码的基础上进行的扩展。
四、字节流缓冲区
我们知道,如果一次读取一次字节,效率是非常慢的,所以我们想到了使用自定义字节数组,一次可以读取多个字节。但是我们都能想到,难道java的开发者难道想不到吗?所以就引入了两个字节流缓冲区类:
- BufferedInputStream 字节输入流缓冲类
- BufferOutputStream 字节输出流缓冲类
这个两个类的构造方法分别需要的参数类型是InputStream对象和OutputStream对象。它们都内置了大小为8192的字节的数组,我们先来看下如何使用,然后再具体分析,还是文件拷贝的例子:
import java.io.BufferedInputStream; import java.io.BufferedOutputStream; import java.io.FileInputStream; import java.io.FileOutputStream; public class BufferedTest { public static void main(String[] args) throws Exception { BufferedInputStream bis = new BufferedInputStream(new FileInputStream("a.txt")); BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("b.txt")); int b; while((b = bis.read()) != -1) { bos.write(b); } bis.close(); bos.close(); } }
有的读者可能会产生这样的疑问,上面的代码不也是一个一个的读取的吗?凭什么它有优势,其实不然,在使用缓冲后,read的时候,是从文件上一次读取8192个字节,保存到输入流缓冲区数组,然后再一个字节一个字节的从输入流缓冲区把字节给java程序,write的时候,java程序再一个字节一个字节传给输出流缓冲区数组,等到输出流缓冲区数组存够8192个字节的时候,再一次性的写出文件上。不使用缓冲区的时候,read方法是一个字节一个字节的从文件读取传递给java程序,write的时候java程序再一个字节一个字节的把字节往文件写。一个字节一个字节的操作相当于是硬盘----java程序(内存)-----硬盘,使用缓冲区后,一个字节一个字节的操作相当于在两个字节流缓冲数组中进行的,输入流缓冲数组(内存)----java程序(内存)----输出流缓冲数组(内存)相当于是在内存中进行传递。我们知道,内存的操作速度是远远大于硬盘的,所以使用缓冲的效率更高。
那么使用缓存和我们自定义字节数组的效率哪个比较高呢?答案是自定义字节数组效率稍微高那么一点点,因为自定义字节数组我们操作是的一个字节数组,而缓冲是两个字节数组,这两个字节数组之间还需要传递字节,但是由于是在内存中进行字节传递,所以效率也是比较高的。这两种方法其实差别不大!
我们会在下篇博文继续讲解字符流的使用!