finally块如何安全关闭资源
时间:2026-04-10 09:33:35 432浏览 收藏
finally块并非自动关闭资源的“保险箱”,它仅保证代码执行,而真正安全释放资源必须手动在其中对每个资源进行判空、单独try-catch包裹close操作,否则一个关闭异常就可能中断整个清理流程,导致连接泄漏或文件句柄耗尽;尽管Java 7+推荐更简洁可靠的try-with-resources,但在维护遗留系统或兼容旧版本时,写对finally仍是防止资源泄露的关键防线——稍有疏忽,看似稳妥的“兜底”反而会掩盖真实错误、放大系统风险。

finally 块真能保证资源关闭吗?
不能。finally 块本身不自动关闭资源,它只是提供一个“无论如何都会执行”的代码位置;是否真正关闭,取决于你在里面写的逻辑是否正确、是否处理了关闭时可能抛出的异常。比如调用 connection.close() 时如果抛出 SQLException,而你没捕获,整个方法会因未处理异常提前退出,后续清理逻辑就断了。
为什么不能只在 try 里 close,也不写 finally?
因为 try 块中一旦发生异常(如 SQL 执行失败、网络中断),控制流会立即跳转到 catch 或直接向外抛出,try 末尾的 close() 根本不会执行。常见错误写法:
try {
Connection conn = DriverManager.getConnection(url);
PreparedStatement stmt = conn.prepareStatement(sql);
stmt.execute();
conn.close(); // ← 这行大概率执行不到
stmt.close();
} catch (SQLException e) {
// 处理异常
}这种写法下,只要 stmt.execute() 抛异常,连接就泄漏了。
finally 中 close 的正确姿势
必须对每个资源的 close() 单独包裹 try-catch,避免一个资源关闭失败阻断其他资源释放。还要注意:null 检查、关闭顺序(通常先子后父,如先 ResultSet 再 Statement 再 Connection)。
close()方法本身可能抛SQLException(JDBC)或IOException(IO 流),必须捕获,且不能简单吞掉——至少要记录日志- 资源变量需声明在
try外(如Connection conn = null;),否则finally无法访问 - Java 7+ 更推荐用 try-with-resources,但 legacy 代码或需兼容老 JDK 时,
finally仍是刚需
示例(JDBC):
Connection conn = null;
PreparedStatement stmt = null;
ResultSet rs = null;
try {
conn = DriverManager.getConnection(url);
stmt = conn.prepareStatement("SELECT * FROM users");
rs = stmt.executeQuery();
// 处理结果
} catch (SQLException e) {
// 处理业务异常
} finally {
if (rs != null) {
try { rs.close(); } catch (SQLException e) { /* log */ }
}
if (stmt != null) {
try { stmt.close(); } catch (SQLException e) { /* log */ }
}
if (conn != null) {
try { conn.close(); } catch (SQLException e) { /* log */ }
}
}流资源(InputStream / OutputStream)的 finally 关闭要点
和 JDBC 类似,但部分流(如 BufferedInputStream)的 close() 会级联关闭底层流,所以不必重复关;不过为清晰起见,仍建议统一判空 + 尝试关闭。关键差异:
- IO 流抛的是
IOException,不是SQLException,别用错 catch 类型 - 某些流(如
ByteArrayInputStream)的close()是空操作,但调用也无害,可统一处理 - 如果流来自
Files.newInputStream()等 NIO 方法,关闭行为一致,仍需判空 +try-catch
示例:
InputStream is = null;
try {
is = new FileInputStream("data.txt");
// 读取...
} finally {
if (is != null) {
try { is.close(); } catch (IOException e) { /* log */ }
}
}多层嵌套资源、异步回调、或资源初始化本身失败(如 new FileInputStream() 抛 FileNotFoundException),都会让 finally 中的判空逻辑更关键——漏掉一个 != null 检查,就可能触发 NullPointerException 掩盖原始错误。
到这里,我们也就讲完了《finally块如何安全关闭资源》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注golang学习网公众号,带你了解更多关于的知识点!
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
150 收藏
-
277 收藏
-
291 收藏
-
122 收藏
-
280 收藏
-
397 收藏
-
315 收藏
-
453 收藏
-
277 收藏
-
233 收藏
-
373 收藏
-
437 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习