Java中级进阶之IO流

 

IO流

 

1.流

概念:流就是一系列的数据

 

1.1 什么是流

 

① 当不同的介质之间有数据交互的时候,JAVA就使用流来实现

② 数据源可以是文件,还可以是数据库,网络甚至是其他的程序

③ 比如读取文件的数据到程序中,站在程序的角度来看,就叫做输入流

输入流: InputStream

输出流:OutputStream

 

1.2 文件输入流

① 建立了一个文件输入流,这个流可以用来把数据从硬盘的文件,读取到JVM(内存)。

 

代码:

1File f = new File("d:/test.txt");
2// 创建基于文件的输入流
3FileInputStream fis = new FileInputStream(f);

 

2. 字节流

概念:用于以字节的形式读取和写入数据『InputStream、OutputStream』

 

2.1 ASCII码 概念

① 所有的数据存放在计算机中都是以数字的形式存放的。 所以字母就需要转换为数字才能够存放。

② 比如:A就对应的数字65,a对应的数字97. 不同的字母和符号对应不同的数字,就是一张码表。

③ ASCII是这样的一种码表。 只包含简单的英文字母、符号、数字等等。

 

2.2 以字节流的形式读取文件内容

① InputStream是字节输入流,同时也是抽象类,只提供方法声明,不提供方法的具体实现。

② FileInputStream 是InputStream子类,以FileInputStream 为例进行文件读取

 

代码:

 

 1//准备文件test.txt 其中的内容是AB,对应的ASCII分别是65 66
 2File f =new File("d:/test.txt");
 3FileInputStream fis =new FileInputStream(f);
 4//创建字节数组,其长度就是文件的长度
 5byte[] all =new byte[(int) f.length()];
 6//以字节流的形式读取文件所有内容
 7fis.read(all);
 8for (byte b : all) {
 9//打印出来是65 66
10System.out.println(b);
11}
12//每次使用完流,都应该进行关闭
13fis.close();

 

2.3 以字节流的形式读取文件内容

 

① OutputStream是字节输出流,同时也是抽象类,只提供方法声明,不提供方法的具体实现。

② FileOutputStream 是OutputStream子类,以FileOutputStream 为例向文件写出数据。

注: 如果文件d:/test2.txt不存在,写出操作会自动创建该文件。但是如果是文件 d:/xyz/test2.txt,而目录xyz又不存在,会抛出异常

 

代码:

 

 1// 准备文件test2.txt其中的内容是空的
 2File f = new File("d:/test2.txt");
 3// 准备长度是2的字节数组,用88,89初始化,其对应的字符分别是X,Y
 4byte data[] = { 88, 89 };
 5// 创建基于文件的输出流
 6FileOutputStream fos = new FileOutputStream(f);
 7// 把数据写入到输出流
 8fos.write(data);
 9// 关闭输出流
10fos.close();
11} catch (IOException e) {
12// TODO Auto-generated catch block
13e.printStackTrace();
14}

 

3. 字符流

概念:Reader字符输入流、Writer字符输出流、专门用于字符的形式读取和写入数据。

 

3.1使用字符流读取文件

代码:

 

 1// 准备文件test.txt其中的内容是AB
 2File f = new File("d:/test.txt");
 3// 创建基于文件的Reader
 4try (FileReader fr = new FileReader(f)) {
 5// 创建字符数组,其长度就是文件的长度
 6char[] all = new char[(int) f.length()];
 7// 以字符流的形式读取文件所有内容
 8fr.read(all);
 9for (char b : all) {
10// 打印出来是A B
11System.out.println(b);
12}
13} catch (IOException e) {
14// TODO Auto-generated catch block
15e.printStackTrace();
16}

 

3.2 使用字符流把字符串写入到文件

代码:

 

 1// 准备文件test.txt
 2File f = new File("d:/test.txt");
 3// 创建基于文件的Writer
 4try (FileWriter fr = new FileWriter(f)) {
 5// 以字符流的形式把数据写入到文件中
 6String data="abcdefg";
 7char[] cs = data.toCharArray();
 8fr.write(cs);
 9} catch (IOException e) {
10// TODO Auto-generated catch block
11e.printStackTrace();
12}

 

4.缓存流

介绍:以介质是硬盘为例,字节流和字符流的弊端:

 

① 在每一次读写的时候,都会访问硬盘。 如果读写的频率比较高的时候,其性能表现不佳。

② 为了解决以上弊端,采用缓存流。

③ 缓存流在读取的时候,会一次性读较多的数据到缓存中,以后每一次的读取,都是在缓存中访问,直到缓存中的数据读取完毕,再到硬盘中读取。

④ 就好比吃饭,不用缓存就是每吃一口都到锅里去铲。用缓存就是先把饭盛到碗里,碗里的吃完了,再到锅里去铲

⑤ 缓存流在写入数据的时候,会先把数据写入到缓存区,直到缓存区达到一定的量,才把这些数据,一起写入到硬盘中去。按照这种操作模式,就不会像字节流,字符流那样每写一个字节都访问硬盘,从而减少了IO操作


4.1 使用缓存流读取数据

代码:

 

 1public static void main(String[] args) {
 2// 准备文件test.txt其中的内容是
 3// sdasd
 4//asdasdasd
 5// sadsfadc
 6File f = new File("d:/test.txt");
 7// 创建文件字符流
 8// 缓存流必须建立在一个存在的流的基础上
 9try (
10FileReader fr = new FileReader(f);
11BufferedReader br = new BufferedReader(fr);
12)
13{
14while (true) {
15// 一次读一行
16String line = br.readLine();
17if (null == line)
18break;
19System.out.println(line);
20}
21} catch (IOException e) {
22// TODO Auto-generated catch block
23e.printStackTrace();
24}

 

4.2 使用缓存流写出数据

概念:PrintWriter 缓存字符输出流, 可以一次写出一行数据

代码:

 

 1// 向文件test2.txt中写入三行语句
 2File f = new File("d:/test2.txt");
 3try (
 4// 创建文件字符流
 5FileWriter fw = new FileWriter(f);
 6// 缓存流必须建立在一个存在的流的基础上
 7PrintWriter pw = new PrintWriter(fw);
 8) {
 9pw.println("garen kill teemo");
10pw.println("teemo revive after 1 minutes");
11pw.println("teemo try to garen, but killed again");
12} catch (IOException e) {
13// TODO Auto-generated catch block
14e.printStackTrace();
15}

 

4.3 flush

概念:有的时候,需要立即把数据写入到硬盘,而不是等缓存满了才写出去。 这时候就需要用到flush

 

代码:

 

 1//向文件test2.txt中写入三行语句
 2File f =new File("d:/test2.txt");
 3//创建文件字符流
 4//缓存流必须建立在一个存在的流的基础上
 5try(FileWriter fr = new FileWriter(f);PrintWriter pw = newPrintWriter(fr);) {
 6pw.println("garen kill teemo");
 7//强制把缓存中的数据写入硬盘,无论缓存是否已满
 8pw.flush();
 9pw.println("teemo revive after 1 minutes");
10pw.flush();
11pw.println("teemo try to garen, but killed again");
12pw.flush();
13} catch (IOException e) {
14// TODO Auto-generated catch block
15e.printStackTrace();
16}
17}


 

5. 数据流

介绍:

 

① 直接进行字符串的读写

② 使用数据流的writeUTF()和readUTF() 可以进行数据的格式化顺序读写

③ 如本例,通过DataOutputStream 向文件顺序写出 布尔值,整数和字符串。 然后再通过DataInputStream 顺序读入这些数据。

④ 注: 要用DataInputStream 读取一个文件,这个文件必须是由DataOutputStream 写出的,否则会出现EOFException,因为DataOutputStream 在写出的时候会做一些特殊标记,只有DataInputStream 才能成功的读取。

 

代码:

 

 1public static void main(String[] args) {
 2write();
 3read();
 4}
 5private static void read() {
 6File f =new File("d:/test.txt");
 7try (
 8FileInputStream fis  = new FileInputStream(f);
 9DataInputStream dis =new DataInputStream(fis);
10){
11boolean b= dis.readBoolean();
12int i = dis.readInt();
13String str = dis.readUTF();
14System.out.println("读取到布尔值:"+b);
15System.out.println("读取到整数:"+i);
16System.out.println("读取到字符串:"+str);
17} catch (IOException e) {
18e.printStackTrace();
19}
20}
21private static void write() {
22File f =new File("d:/test.txt");
23try (
24FileOutputStream fos  = newFileOutputStream(f);
25DataOutputStream dos =newDataOutputStream(fos);
26){
27dos.writeBoolean(false);
28dos.writeInt(200);
29dos.writeUTF("123 wew d wd s ");
30} catch (IOException e) {
31e.printStackTrace();
32}
33}

6. 对象流

 

6.1 介绍

 

① 对象流指的是可以直接把一个对象以流的形式传输给其他的介质,比如硬盘

② 一个对象以流的形式进行传输,叫做序列化。 该对象所对应的类,必须是实现Serializable接口

 

6.2序列化一个对象

①创建一个H对象,设置其名称为g。

②把该对象序列化到一个文件g.txt。

然后再通过序列化把该文件转换为一个H对象。

注:把一个对象序列化有一个前提是:这个对象的类,必须实现了Serializable接口。

 

代码:

 1public class Hero implements Serializable {
 2//表示这个类当前的版本,如果有了变化,比如新设计了属性,就应该修改这个版本号
 3private static final long serialVersionUID = 1L;
 4public String name;
 5public float hp;
 6}
 7public class TestStream {
 8public static void main(String[] args) {
 9//创建一个Hero garen
10//要把Hero对象直接保存在文件上,务必让Hero类实现Serializable接口
11Hero h = new Hero();
12h.name = "garen";
13h.hp = 616;
14//准备一个文件用于保存该对象
15File f =new File("d:/garen.lol");
16try(
17//创建对象输出流
18FileOutputStream fos = new FileOutputStream(f);
19ObjectOutputStream oos =new ObjectOutputStream(fos);
20//创建对象输入流
21FileInputStream fis = new FileInputStream(f);
22ObjectInputStream ois =new ObjectInputStream(fis);
23) {
24oos.writeObject(h);
25Hero h2 = (Hero) ois.readObject();
26System.out.println(h2.name);
27System.out.println(h2.hp);
28} catch (IOException e) {
29// TODO Auto-generated catch block
30e.printStackTrace();
31} catch (ClassNotFoundException e) {
32// TODO Auto-generated catch block
33e.printStackTrace();
34}
35}
36}

 

2. 关闭流的方式

概念:所有的流,无论是输入流还是输出流,使用完毕之后,都应该关闭。 如果不关闭,会产生对资源占用的浪费。 当量比较大的时候,会影响到业务的正常开展。

 

2.1 在try中关闭

弊端:如果文件不存在,或者读取的时候出现问题而抛出异常,那么就不会执行这一行关闭流的代码,存在巨大的资源占用隐患。

 

代码:

 

 1try {
 2File f = new File("d:/test.txt");
 3FileInputStream fis = new FileInputStream(f);
 4byte[] all = new byte[(int) f.length()];
 5fis.read(all);
 6for (byte b : all) {
 7System.out.println(b);
 8}
 9// 在try 里关闭流
10fis.close();
11} catch (IOException e) {
12e.printStackTrace();
13}

 

2.2  在finally中关闭「标准的关闭流的方式」

介绍:

①  首先把流的引用声明在 try 的外面,如果声明在 try 里面,其作用域无法抵达finally .

②  在 finally 关闭之前,要先判断该引用是否为空

③ 关闭的时候,需要再一次进行 try catch 处理

 

代码:

 

 1File f = new File("d:/test.txt");
 2FileInputStream fis = null;
 3try {
 4fis = new FileInputStream(f);
 5byte[] all = new byte[(int) f.length()];
 6fis.read(all);
 7for (byte b : all) {
 8System.out.println(b);
 9}
10} catch (IOException e) {
11e.printStackTrace();
12} finally {
13// 在finally 里关闭流
14if (null != fis)
15try {
16fis.close();
17} catch (IOException e) {
18// TODO Auto-generated catch block
19e.printStackTrace();
20}
21}

 

2.3 使用 try() 的方式「把流定义在 try() 里,try 、catch 或者 finally 结束的时候,会自动关闭」

 

代码:

 

 1File f = new File("d:/test.txt");
 2//把流定义在try()里,try,catch或者finally结束的时候,会自动关闭
 3try (FileInputStream fis = new FileInputStream(f)) {
 4byte[] all = new byte[(int) f.length()];
 5fis.read(all);
 6for (byte b : all) {
 7System.out.println(b);
 8}
 9} catch (IOException e) {
10e.printStackTrace();
11}

 

 

 

你可能感兴趣的:(java)