【Node.js】为什么数据库连接总是中断?

Node.js 中使用 import mysql from 'mysql2/promise'; 本身并不会 直接导致数据库停止工作。但在长时间运行的 Node.js 程序中,如果代码编写不当(特别是连接管理方面),确实会导致数据库连接耗尽、超时或导致 Node.js 应用崩溃。

以下是几个可能导致"数据库停止工作"或"连接失效"的关键原因及解决方法:

1. 连接泄露(最常见原因)

如果你每次执行查询都创建一个新的连接(createConnection)而没有手动关闭它(.end()),或者在高并发下没有正确释放连接,数据库很快就会达到 max_connections(最大连接数)限制。

  • 后果: 数据库拒绝新的连接,报错 Too many connections
  • 解决方法: 使用 连接池 (Connection Pool)

2. 空闲超时 (Idle Timeout)

MySQL 服务器有一个参数叫 wait_timeout(通常默认为 8 小时)。如果一个连接长时间没有任何操作,MySQL 服务器会强制断开这个连接。

  • 后果: 当你的程序在长时间休眠后再次使用该连接时,会报错 Can't add new command when connection is in closed stateread ECONNRESET
  • 解决方法: 连接池会自动处理这个问题,它会检查连接的有效性并在需要时重新创建连接。

3. 未捕获的 Promise 异常

使用 mysql2/promise 时,如果数据库查询报错(如 SQL 语法错误、连接断开)而你没有使用 try...catch 捕获异常,Node.js 进程可能会触发 unhandledRejection 甚至崩溃。

  • 后果: Node.js 应用停止运行,看起来像是数据库"坏了"。
  • 解决方法: 始终使用 try...catch 包裹异步操作。

最佳实践:如何正确长期运行

为了保证 Node.js 长时间运行且数据库稳定,你应该按照以下方式编写代码:

推荐:使用连接池 (Connection Pool)

连接池会自动管理连接的创建、复用和销毁,是长生命周期应用(如 Web 服务器)的标准做法。

javascript 复制代码
import mysql from 'mysql2/promise';

// 1. 创建连接池(只需创建一次,全局复用)
const pool = mysql.createPool({
  host: 'localhost',
  user: 'root',
  password: 'password',
  database: 'test',
  waitForConnections: true,
  connectionLimit: 10, // 最大连接数
  queueLimit: 0,
  enableKeepAlive: true, // 保持心跳,防止被服务器断开
  keepAliveInitialDelay: 10000
});

async function queryData() {
  try {
    // 2. 直接从池中执行(内部会自动获取、执行、释放连接)
    const [rows] = await pool.query('SELECT * FROM users WHERE id = ?', [1]);
    return rows;
  } catch (error) {
    console.error('数据库查询出错:', error);
  }
}

// 周期性运行,不会导致数据库停止
setInterval(async () => {
  const data = await queryData();
  console.log('查询结果:', data);
}, 60000); 

检查清单(防止失效):

  1. 使用 Pool 而不是 Connection: 永远不要在循环或频繁触发的函数中调用 mysql.createConnection()
  2. 启用 KeepAlive:createPool 配置中加入 enableKeepAlive: true。这会让驱动程序定期发送微小的包,防止防火墙或 MySQL 服务器因为该连接"太安静"而将其切断。
  3. 异常处理: 确保所有的 await 都在 try...catch 中,防止程序因为单个查询失败而挂掉。
  4. 不要手动关闭池: 除非你的程序要完全关闭,否则不要调用 pool.end()

总结

单纯的 import 没问题,但要确保使用 createPool 而不是 createConnection。只要使用了连接池并妥善处理了 try...catch,你的 Node.js 程序跑一年数据库也不会因为连接问题停止工作。

相关推荐
counterxing7 小时前
Agent 跑起来之后,难的是复用、观测和评测
node.js·agent·ai编程
星星也在雾里8 小时前
PgBouncer 解决 PostgreSQL 连接数超限 + 可视化监控
数据库·postgresql
雨辰AI10 小时前
SpringBoot3 + 人大金仓读写分离 + 分库分表 + 集群高可用 全栈实战
java·数据库·mysql·政务
长城202410 小时前
关于MySql的ONLY_FULL_GROUP_BY问题
数据库·mysql·聚合列
常常有10 小时前
MySQL 底层执行原理:输入SQL语句到两阶段提交
数据库·sql·mysql
Mr. zhihao11 小时前
深入解析redis基本数据结构
数据结构·数据库·redis
m0_7488394911 小时前
利用天正暖通CAD快速掌握风管数量统计的方法
数据库
随身数智备忘录11 小时前
什么是设备管理体系?设备管理体系包含哪些核心模块?
网络·数据库·人工智能
海市公约12 小时前
MySQL更新语句执行全流程:从Buffer Pool修改到二阶段提交
数据库·mysql·binlog·innodb·undo log·二阶段提交·update执行原理
颂love12 小时前
MySQL的执行流程
android·数据库·mysql