Overview

Closeable和AutoCloseable都是接口,且都只定义了一个close()方法。

Closeable: 定义于 java.io包中,于JDK5添加;

AutoCloseable: 定义于java.lang包中, 于JDK7添加;

AutoCloseable.java

public interface AutoCloseable{

void close() throws Exception;

}

从JDK7开始,Closeable扩展了AutoCloseable

Closeable.java

public interface Closeable extends AutoCloseable {

public void close() throws IOException;

}

因此从JDK7开始,Closeable和AutoCloseable具有相同的功能,因此下面只介绍AutoCloseable。

AutoCloseable 是什么

AutoCloseable是一个接口,只定义了一个close()方法。接口是什么?是为了给某些类实现的,它强制要求这个类提供接口定义的方法。所有实现AutoCloseable接口的类都必须提供close()方法。

那什么样的类需要实现AutoCloseable(Closeable)接口呢? 看看它定义的方法, close()。看名字就知道所有实现这个接口的类都必须提供关闭的方法,关闭的自然是资源, 比如各类输入输出流资源, Socket等。

常见的实现了AutoCloseable接口的类有:

BufferedInputStream, BufferedOutputStream

BufferedReader, BufferedWriter

FileInputStream, FileOutputStream

FileReader, FileWriter

InputStream, OutputStream

PrintWriter, PrintStream

Reader, Writer

Scanner, SSLServerSocker, SSLSocket等等等等

为什么要使用AutoCloseable

在JDK1.7以前,如果我们要用到上面的这些资源类,因为没有他们没有实现AutoCloseable接口, 正常的做法是需要使用try-finally结构, 在finally中手动的释放资源:

//JDK1.7之前,释放资源方式

FileInputStream fileInputStream = null;

try {

fileInputStream = new FileInputStream("");

} catch (FileNotFoundException e) {

e.printStackTrace();

} finally {

try {

fileInputStream.close();

} catch (IOException e) {

e.printStackTrace();

}

}

这么做的坏处是:

1、自己要手动写代码做关闭的逻辑;

2、有时候还会忘记关闭一些资源;

3、关闭代码的逻辑比较冗长,不应该是正常的业务逻辑需要关注的;

为了解决需要手动释放资源的问题, 在JDK1.7中引入了AutoCloseable接口,凡是实现了这个接口的资源类在使用的时候采用另外一种语法,可以使得免于手动关闭这些资源类。

怎么使用AutoCloseable

使用已经实现了AutoCloseable接口的资源类

对于实现了AutoCloseable接口的类的实例,将其放到try后面的小括号中,这种语法我们称之为:带资源的try语句,在try结束的时候,会自动将这些资源关闭(调用close方法)。

//1.7之后,只要实现了AutoCloseable接口

try (FileInputStream fileInputStream2 = new FileInputStream("");

Reader rd = new Reader()) {

} catch (FileNotFoundException e) {

e.printStackTrace();

} catch (IOException e) {

e.printStackTrace();

注意这里申请资源的语句放在了紧跟着try的小括号中,这属于JDK1.7的新语法。只要用这样的语法申请实现了AutoCloseable接口的资源类,就无需使用finally块手动释放这些资源。

带资源的try语句的3个关键点:

由带资源的try语句管理的资源必须是实现了AutoCloseable接口的类的对象。

在try代码中声明的资源被隐式声明为final。

通过使用分号分隔每个声明可以管理多个资源。

定义自己的资源类,实现AutoCloseable接口

对于自己定义的资源类,如果也想像FileInputStream一样在使用的时候用带资源的try语句实现自动释放,做法就很显然: 让这个类去实现AutoCloseable接口并重写close()方法, 在重写的close()方法中提供释放资源的语句。

class MyResource implements AutoCloseable {

/**

对这个接口的实现,规范强烈建议close()是幂等的,也就是说多次调用close()方法和一次调用的结果是一样的。

*/

@Override

public void close() throws Exception {

//添加释放资源的语句

System.out.println("close resource");

}

public void readResource() {

//提供资源的方法

System.out.println("read resource");

}

}

这样定义了MyResource类后,就可以像如下方式使用它:

//示例,使用自己定义的资源类,它实现了AutoCloseable接口

try (MyResource myResource = new MyResource()) {

myResource.readResource();

myResource2.readResource();

} catch (Exception e) {

e.printStackTrace();

}

注意

虽然在JDK1.7之后AutoCloseable和Closeable具有一样的功能,但一般实现自己的资源类的时候我们让它实现AutoCloseable类。Closeable比AutoCloseable更早出现(JDK5),在JDK7之前它提供了一套关闭资源的方法,但是后来的AutoCloseable可以用更好的方法来关闭资源,于是从JDK7开始弃用了之前Closeable接口关闭资源的方法,而让它去扩展AutoCloseable接口,这样先前实现Closeable类的方法也可以过度到AutoCloseable的解决方案中来。

需要注意的是AutoCloseable定义在java.lang包中,这个包在java启动时自动加载到环境,所以使用AutoCloseable时无需额外引包;而Closeable定义在java.io中,如果要使用这个接口需要手动引入java.io.Closeable;