使用Java开发业务时,常常需要处理资源,这是很常见的需求。
在Java 7以前,处理(关闭)资源是这样的:
public static int getAccountStatusCodeFromDataStore_traditional(String accountId) throws SQLException { String accountStatusCodeQuery = getAccountStatusCodeQuery(accountId); Statement statement = null; ResultSet resultSet = null; try { statement = createStatementFromConnection(); resultSet = statement.executeQuery(accountStatusCodeQuery); return getAccountStatusCodeFromResultSet(resultSet); } finally { if (resultSet != null) resultSet.close(); if (statement != null) statement.close(); } }
开发人员必须关闭创建的所有资源,否则会导致资源泄漏。
而在Java 7中,引入了try-with-resources的新方法,可以在try-catch块中使用的正确顺序自动处理资源的关闭,比如:
public static int getAccountStatusCodeFromDataStore_java7(String accountId) throws SQLException { String accountStatusCodeQuery = getAccountStatusCodeQuery(accountId); try (Statement statement = createStatementFromConnection(); ResultSet resultSet = statement.executeQuery(accountStatusCodeQuery)) { return getAccountStatusCodeFromResultSet(resultSet); } }
在此示例中,可以看到代码更简洁了,整体可读性提高了,它实现了资源的自动管理。我们可以在try-with-resources语句中拥有多个资源,且多个资源的声明之间应该用分号分隔。当这些资源在自动关闭时,也会保持声明的反向逻辑顺序依次关闭(最后声明的资源将首先关闭)。
如果这里要抛出异常,try块的异常会会压制try-with-resources块的异常。如果确实有需要,可以通过从try块抛出的异常中调用Throwable.getSuppressed方法来检索被try块抑制的异常。
另外,try-with-resources语句也可以有catch和finally块。在声明的资源被关闭后会运行任何catch或finally块。
而到了Java 9时代,对于try-with-resources的资源处理,Java 9中引入了更简洁的版本。如果开发者已经将资源声明为final或effective final类型,那么可以在try-with-resources中直接使用它们而无需创建任何新变量。这使得我们可以进一步利用自动资源管理。上面的代码现在可使用更简洁的try-with-resources实现,如下:
public static int getAccountStatusCodeFromDataStore_java9(String accountId) throws SQLException { String accountStatusCodeQuery = getAccountStatusCodeQuery(accountId); // 明确声明final final Statement statement = createStatementFromConnection(); // effective final ResultSet resultSet = statement.executeQuery(accountStatusCodeQuery); try (statement; resultSet) { return getAccountStatusCodeFromResultSet(resultSet); } }
可见,代码的易读性提高了。
其实大多数的资源类在背后实现了AutoCloseable或Closeable接口,因此与try-with-resources语句协同工作才实现了自动资源管理。如果我们处理的资源没有实现AutoCloseable或Closeable接口,那么就必须遵循传统的方法来关闭资源。