java7 之前关闭资源都需要在 finally 中明确指出要关闭的流,比较繁琐 java7 之后就可以使用 try-with-resources 来实现了

原始关闭资源

  1. public void test() throw IOException {
  2. FileInputStream inputStream = null;
  3. try {
  4. inputStream = new FileInputStream(new File("/tmp.txt"));
  5. } finally {
  6. if (inputStream != null) {
  7. inputStream.close();
  8. }
  9. }
  10. }

上面代码中的程序可能会抛出异常。try 语句块中有 1 个地方能抛出异常,finally 语句块中有一个地方会能出异常。
不论 try 语句块中是否有异常抛出,finally 语句块始终会被执行。这意味着,不论try语句块中发生什么,InputStream 都会被关闭,或者说都会试图被关闭。如果关闭失败,InputStream.close()方法也可能会抛出异常。

假设 try 语句块抛出一个异常,然后 finally 语句块被执行。同样假设 finally 语句块也抛出了一个异常。那么哪个异常会根据调用栈往外传播?
即使 try 语句块中抛出的异常与异常传播更相关,最终还是 finally 语句块中抛出的异常会根据调用栈向外传播。

Java7关闭资源

  1. public void test2() throws IOException {
  2. try(FileInputStream inputStream = new FileInputStream(new File("/tmp.txt"))
  3. ) {
  4. // do something
  5. }
  6. }

注意方法中的第一行:

  1. try(FileInputStream inputStream = new FileInputStream(new File("/tmp.txt")))

这就是try-with-resource 结构的用法。FileInputStream 类型变量就在try关键字后面的括号中声明。而且一个FileInputStream 类型被实例化并被赋给了这个变量。
当try语句块运行结束时,FileInputStream 会被自动关闭。这是因为FileInputStream 实现了java中的java.lang.AutoCloseable接口。所有实现了这个接口的类都可以在try-with-resources结构中使用。

当try-with-resources结构中抛出一个异常,同时FileInputStreami被关闭时(调用了其close方法)也抛出一个异常,try-with-resources结构中抛出的异常会向外传播,而FileInputStreami被关闭时抛出的异常被抑制了。这与文章开始处利用旧风格代码的例子(在finally语句块中关闭资源)相反。

声明多个资源

可以在块中使用多个资源而且这些资源都能被自动地关闭。

  1. public void test2() throws IOException {
  2. try(FileInputStream inputStream = new FileInputStream(new File("/tmp.txt"));
  3. Reader reader = new BufferedReader(new InputStreamReader(inputStream, "utf-8"), 10240)
  4. ) {
  5. // do something
  6. }
  7. }

上面的例子在 try 关键字后的括号里创建了两个资源—— FileInputStream 和 BufferedInputStream。
当程序运行离开 try 语句块时,这两个资源都会被自动关闭。
这些资源将按照他们被创建顺序的逆序来关闭。首先 BufferedInputStream 会被关闭,然后 FileInputStream 会被关闭。

自定义AutoClosable 实现

AutoClosable 接口仅仅有一个方法,接口定义如下:

  1. public interface AutoCloseable {
  2. void close() throws Exception;
  3. }

任何实现了这个接口的方法都可以在try-with-resources结构中使用。下面是一个简单的例子:

  1. public class MyAutoClosable implements AutoCloseable {
  2. public void doIt() {
  3. System.out.println("MyAutoClosable doing it!");
  4. }
  5. @Override
  6. public void close() throws Exception {
  7. System.out.println("MyAutoClosable closed!");
  8. }
  9. }

doIt() 是方法不是 AutoClosable 接口中的一部分,之所以实现这个方法是因为我们想要这个类除了关闭方法外还能做点其他事。

下面是MyAutoClosable 在try-with-resources结构中使用的例子:

private static void myAutoClosable() throws Exception {
    try(MyAutoClosable myAutoClosable = new MyAutoClosable()){
        myAutoClosable.doIt();
    }
}

当方法myAutoClosable.doIt()被调用时,下面是打印到System.out的输出:

MyAutoClosable doing it!
MyAutoClosable closed!

通过上面这些你可以看到,不论 try-catch 中使用的资源是自己创造的还是 java 内置的类型,try-with-resources都是一个能够确保资源能被正确地关闭的强大方法。