在 Java 应用与 Oracle 数据库的交互过程中,“连接池泄漏”这一术语往往令人高度警惕,但更为棘手的是:即便你的代码已经规范地调用了 close() 方法,数据库连接数依然可能持续攀升。此现象背后的根本原因,远不止“忘记调用 close”这么简单。
从本质上讲,Connection.close() 仅完成了一项操作:在 JDBC 驱动层面将该连接标记为“可复用”。至于该连接是否真正归还至连接池、归还后能否被再次成功使用,完全取决于 HikariCP 或 Druid 等连接池的具体实现机制,以及 Oracle 网络层的实时状态。例如,TNS 层遗留的半开连接——当网络中断而驱动未能感知到断连时,close() 可能默默阻塞甚至执行失败。再如,HikariCP 检测到连接异常时,会直接丢弃而非回收,从而造成“已 close 但活跃数未下降”的假象。更隐蔽的是,自 Oracle JDBC 驱动 12.2.0.1+ 版本起,implicitCachingEnabled 默认处于开启状态。若前一个业务未调用 preparedStatement.clearCache(),复用该连接时隐式缓存将持续膨胀,最终触发 ORA-01000: maximum open cursors exceeded 错误。不仅如此,连接池的超时参数若与 Oracle 网络层的 SQLNET.EXPIRE_TIME 不匹配,连接在池中“假活”数小时的情况也屡见不鲜。

高负载场景下,泄漏问题的放大效应
在低流量时段系统运行平稳,一旦请求量激增,那些潜在的隐性缺陷便会集中爆发。以下三个关键放大器值得特别关注:
第一,超时参数的错位配置。若 connection-timeout 设置过小(例如 1000 毫秒),大量线程将阻塞在“获取连接”阶段,监控显示活跃连接数并未上涨,但应用整体已陷入僵局。反之,若设置过大(例如 60000 毫秒),故障感知将严重滞后,DBA 端看到会话数急剧飙升,却难以迅速定位根源。
第二,校验机制的缺失。禁用 testOnBorrow 后,若未配置 connection-init-sql 或 keepaliveTime,连接在复用前将缺少轻量级校验。直到首次执行 SQL 时,TNS 错误才会暴露,但此时已进入业务逻辑,finally 块中的 close() 很可能因异常而被跳过。
第三,废弃配置的遗留隐患。Druid 的 removeAbandonedOnBorrow=true 早已建议退役,但若旧配置仍残留,它会在借出连接时强制回收“疑似泄漏”的连接。在高负载下判断容易失准,频繁回收健康连接将引发雪崩式的重连和游标泄漏问题。
别再猜测,直接查询数据库
排查连接池泄漏问题时,不要仅盯着应用日志,数据库侧的真实连接状态才是最可靠的证据。直接查询 v$session 视图,定位那些 last_call_et 大于 300 秒且状态仍为 ACTIVE 的会话——这正是典型的“假活”连接。再交叉比对 v$open_cursor 中各个会话的游标数量,若某几个 sid 的游标数持续超过 300,基本可锁定对应的应用线程未正确清理 PreparedStatement。
同时检查 v$process 的 pga_used_mem 和 v$session 的 program 字段,确认高内存占用进程是否来自 Java 应用。更高效的方法是结合 AWR 报告,关注“SQL ordered by Parse Calls”指标。若某条简单 SQL 的解析次数远高于执行次数,说明应用在反复创建新的 PreparedStatement 而非复用——这通常是隐式缓存失效或未使用预编译语句缓存的明确信号。
三处关键配置硬伤,必须整改
以下几条并非“建议”,而是 Oracle 生产环境连接池的底线配置要求。
- HikariCP 参数:
connection-timeout=30000(建议小于等于 DBA 设置的SQLNET.EXPIRE_TIME * 1000),validation-timeout=3000,keepaliveTime=300000(5 分钟心跳),禁用testOnBorrow,启用connection-init-sql,例如ALTER SESSION SET CURRENT_SCHEMA=xxx。 - Oracle JDBC URL:必须显式关闭隐式缓存,添加
?implicitCachingEnabled=false&oracle.jdbc.useFetchSizeWithLongColumn=true,避免驱动层缓存干扰连接池管理。 - 代码写法:所有
Connection获取必须包裹在try-with-resources语句中,禁止在 if/else 分支中手动编写close()。若不得已使用finally,务必加入双重检查:connection != null && !connection.isClosed()。
最容易被忽视也最为关键的一点是:Oracle 连接池泄漏往往并非单一代码 bug 所致,而是连接池参数、驱动行为、数据库网络策略、应用编码习惯四者错位叠加的最终结果。因此,排查问题时,不要急于翻阅 Java 堆栈。先去数据库执行一遍 v$session 和 v$open_cursor 查询——数据库比你的应用更清楚,到底是谁占用了连接却未归还。
