7

我正在查看 Java 中的 try-with-resources 示例,我了解以下内容:

try (Connection conn = DriverManager.getConnection(url, user, pwd);
     Statement stmt = conn.createStatement();
     ResultSet rs = stmt.executeQuery(query);) {
  ...
}

所以,关闭的顺序是:

rs.close();
stmt.close();
conn.close();

这是完美的,因为一个连接有一个语句,一个语句有一个结果集。

但是,在以下示例中,我认为关闭的顺序与预期相反:

示例 1:

try (FileReader fr = new FileReader(file);
     BufferedReader br = new BufferedReader(fr)) {
  ...
}

关闭顺序为:

br.close();
fr.close();

示例 2:

try (FileOutputStream fos = new FileOutputStream("testSer.ser");
    ObjectOutputStream oos = new ObjectOutputStream(fs);) {
    ...
}

关闭顺序为:

oos.close();
fos.close();

这些例子正确吗?我认为这些例子中的结束应该是不同的,因为:

  1. 在示例 1 中,BufferedReader 具有 FileReader。
  2. 在示例 2 中,ObjectOutputStream 具有 FileOutputStream。
4

2 回答 2

7

顺序是相同的:它总是与指定资源的顺序相反。来自JLS

资源以与它们初始化时相反的顺序关闭。

但是,如果后面指定的资源本身调用了close()前面指定的资源的方法(例如BufferedReaderand的情况ObjectOutputStream),它们可能看起来不像预期的顺序发生(并且close()会被多次调用)。

于 2018-10-08T09:48:08.230 回答
0

当我说“一个连接有一个语句并且一个语句有一个结果集”时,我认为我是不对的。也许它是相反的“一个结果集有一个语句,一个语句有一个连接”,或者至少“一个结果集是由一个语句创建的,一个语句是由一个连接创建的”。

所以我认为:

try (Parent parent = new Parent();
     Child child = parent.createChild();) {
    ...
}

相当于:

try (Parent parent = new Parent();
     Child child = new Child(parent);) {
    ...
}
于 2018-10-08T20:46:34.233 回答