> 所有的流,无论是输入流还是输出流,使用完毕之后,都应该关闭。 如果不关闭,会产生对资源占用的浪费。 当量比较大的时候,会影响到业务的正常开展。
# 在try中关闭
在try的作用域里关闭文件输入流,在前面的示例中都是使用这种方式,这样做有一个弊端;
如果文件不存在,或者读取的时候出现问题而抛出异常,那么就不会执行这一行关闭流的代码,存在巨大的资源占用隐患。 **不推荐使用**
```
package com.dodoke.util;
import java.io.File;
import java.io.FileInputStream;
public class TestSteam3 {
public static void main(String[] args) {
try {
// 准备文件lol.txt其中的内容是AB,对应的ASCII分别是65 66
File f = new File("d:/log.txt");
// 创建基于文件的输入流
FileInputStream fis = new FileInputStream(f);
// 创建字节数组,其长度就是文件的长度
byte[] all = new byte[(int) f.length()];
// 以字节流的形式读取文件的所有内容
fis.read(all);
for (byte b : all) {
System.out.println(b);
}
// 每次使用完流,都应该进行关闭
fis.close();
} catch (Exception e) {
e.printStackTrace();
}
}
}
```
# 在finally中关闭
这是标准的关闭流的方式
1. 首先把流的引用声明在try的外面,如果声明在try里面,其作用域无法抵达finally.
2. 在finally关闭之前,要先判断该引用是否为空
3. 关闭的时候,需要再一次进行`try catch`处理
这是标准的严谨的关闭流的方式,但是看上去很繁琐,所以写不重要的或者测试代码的时候,都会采用上面的有**隐患try**的方式,因为不麻烦~
```
package com.dodoke.util;
import java.io.File;
import java.io.FileInputStream;
public class TestSteam3 {
public static void main(String[] args) {
FileInputStream fis = null;
try {
// 准备文件lol.txt其中的内容是AB,对应的ASCII分别是65 66
File f = new File("d:/log.txt");
// 创建基于文件的输入流
fis = new FileInputStream(f);
// 创建字节数组,其长度就是文件的长度
byte[] all = new byte[(int) f.length()];
// 以字节流的形式读取文件的所有内容
fis.read(all);
for (byte b : all) {
System.out.println(b);
}
} catch (Exception e) {
e.printStackTrace();
} finally {
// 在finally 里关闭流
if (null != fis) {
try {
fis.close();
} catch (Exception e) {
e.printStackTrace();
}
}
}
}
}
```
# 使用try()的方式
把流定义在`try()`里,try,catch或者finally结束的时候,会自动关闭
这种编写代码的方式叫做 **`try-with-resources`**, 这是从JDK7开始支持的技术
所有的流,都实现了一个接口叫做 **`AutoCloseable`**,任何类实现了这个接口,都可以在`try()`中进行实例化。 并且在try, catch, finally结束的时候自动关闭,回收相关资源。
```
package com.dodoke.util;
import java.io.File;
import java.io.FileInputStream;
public class TestSteam3 {
public static void main(String[] args) {
// 准备文件lol.txt其中的内容是AB,对应的ASCII分别是65 66
File f = new File("d:/log.txt");
try (FileInputStream fis = new FileInputStream(f);) {
// 创建字节数组,其长度就是文件的长度
byte[] all = new byte[(int) f.length()];
// 以字节流的形式读取文件的所有内容
fis.read(all);
for (byte b : all) {
System.out.println(b);
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
```