登录
首页 >  文章 >  java教程

try-with-resources自动关闭资源使用教程

时间:2026-02-03 20:43:55 460浏览 收藏

从现在开始,我们要努力学习啦!今天我给大家带来《try with resources自动关闭资源详解》,感兴趣的朋友请继续看下去吧!下文中的内容我们主要会涉及到等等知识点,如果在阅读本文过程中有遇到不清楚的地方,欢迎留言呀!我们一起讨论,一起学习!

只有实现 AutoCloseable 或 Closeable 接口的类(如 FileInputStream、BufferedReader、JDBC 资源等)才能用于 try-with-resources;close() 在块结束时按声明逆序调用,异常会被压制,且不保证关闭成功或及时。

在Java中try with resources如何自动关闭资源_Java资源管理解析

Java 的 try-with-resources 会自动调用资源的 close() 方法,但前提是该资源实现了 AutoCloseable 接口——不是所有“带 close 的类”都满足这个条件,也不是所有 close 调用都会如预期执行。

哪些类能放进 try-with-resources 语句里

只有声明实现 AutoCloseable(或其子接口 Closeable)的类才能用于 try-with-resources。JDK 中常见支持的包括:

  • FileInputStreamBufferedReaderConnection(JDBC)、StatementResultSet
  • ZipInputStreamObjectOutputStreamScanner(构造参数为可关闭资源时)
  • 自定义类必须显式 implements AutoCloseable,且 close() 方法不能抛出受检异常(或只能抛 Exception 及其子类,但推荐不抛或只抛 IOException

反例:ThreadTimerSocket(注意:它实现了 Closeable,但实际使用中需谨慎,因为 close() 可能中断读写流逻辑)。

close() 是在什么时候被调用的

try-with-resources 中的资源会在语句块结束时(无论正常退出还是因异常退出)被按**声明逆序**自动调用 close()。这和手动 finally 块中关闭的顺序相反。

例如:

try (BufferedReader br = new BufferedReader(...);
     PrintWriter pw = new PrintWriter(...)) {
    // ...
}

等价于先调 pw.close(),再调 br.close()。这点对依赖顺序的资源(如嵌套流)很关键——比如 BufferedInputStream 包裹 FileInputStream,必须先关外层再关内层,否则缓冲可能丢失数据。

异常压制(suppression)机制怎么影响错误排查

如果 try 块抛出异常,且 close() 也抛异常,后者会被“压制”,仅作为 getSuppressed() 返回的异常列表存在,不会中断主异常流程。

这意味着你可能根本看不到 close() 失败的真实原因:

  • 主异常是 NullPointerException,但 close() 实际抛了 IOException → 后者被压制
  • 若想捕获所有问题,需显式检查:e.getSuppressed() 并遍历打印
  • 某些框架(如 Spring JDBC)会主动吞掉 close() 异常,导致连接泄漏却无日志提示

一个典型陷阱:Connection.close() 在连接已断开时可能抛 SQLException,但它常被压制,掩盖了网络或连接池配置问题。

try-with-resources 不等于绝对安全

自动调用 close() 只解决“是否调用”的问题,不解决“是否成功”或“是否及时”的问题:

  • 资源未实现 AutoCloseable → 编译失败,但有人会绕过(比如把资源声明在 try 外,再手动 close,这就失去语法保障
  • close() 方法内部空实现或仅做日志 → 看似关闭,实则资源(如文件句柄、数据库连接)仍在占用
  • 资源被提前 null 或重新赋值 → 编译器仍会尝试调用 close,但对象可能已不可达,引发 NullPointerException
  • 多线程共享同一资源实例 → try-with-resources 关闭后其他线程继续使用,抛 IllegalStateException 或静默失败

最易被忽略的一点:流式 API 链式调用(如 Files.newBufferedReader(Paths.get("x")).lines().forEach(...))中,BufferedReader 没有被显式绑定到 try 资源列表,也就不会自动关闭——必须拆成独立声明。

以上就是《try-with-resources自动关闭资源使用教程》的详细内容,更多关于的资料请关注golang学习网公众号!

前往漫画官网入口并下载 ➜
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>