0

0

如何正确使用 pg-promise 实现事务内批量数据库操作并避免未捕获异常

心靈之曲

心靈之曲

发布时间:2026-01-05 20:15:08

|

287人浏览过

|

来源于php中文网

原创

如何正确使用 pg-promise 实现事务内批量数据库操作并避免未捕获异常

本文详解 pg-promise 中事务(`db.tx`)与 promise 批量执行的正确模式,指出 `t.batch()` 已废弃,推荐使用 `await` 串行调用或 `promise.all()` 并行执行,并强调统一错误处理与上下文传递(如传入 `t`)是避免“uncaught exception”的关键。

在使用 pg-promise 构建健壮的数据库事务逻辑时,一个常见陷阱是:将已执行(即已触发)的 Promise 直接传入 t.batch() 或封装函数中,导致错误无法被事务上下文捕获。你遇到的“服务器崩溃 + Uncaught Exception”正是这一问题的典型表现——根本原因在于:Promise 一旦创建即开始执行,若其内部抛出异步错误而未被及时 .catch(),且又未被事务作用域包裹,则会逸出到全局,触发 Node.js 的 unhandledRejection 事件,最终导致进程崩溃(尤其在未监听该事件时)

✅ 正确做法:延迟执行 + 统一错误处理

核心原则是:所有数据库操作必须在事务回调函数内部、由 t(事务对象)发起,而非提前在外部构造 Promise。这意味着:

  • ❌ 错误示范(提前执行):

    // ⚠️ addToColumn 立即执行!返回的是一个正在运行的 Promise
    const queries = [
      addToColumn('envelopes', 'budget', id1, -amt, t), // 此处 t 未定义 → 实际调用 db.one → 脱离事务!
      addToColumn('envelopes', 'budget', id2, amt, t)
    ];
    return t.batch(queries); // 传入的是“已启动”的 Promise,错误无法被 t 捕获
  • ✅ 正确示范(延迟执行,由 t 驱动):

    WPS灵犀
    WPS灵犀

    WPS灵犀是WPS推出的一款AI智能办公和学习助手

    下载
    async function transferEnvelopeBudgetByIds(req, res, next) {
      try {
        const result = await db.tx(async t => {
          // ✅ 所有操作均在 t 作用域内按需执行,自动受事务保护
          const from = await t.one(
            'UPDATE ${table:name} SET ${column:name} = ${column:name} + ${amount:csv} WHERE id = ${id:csv} RETURNING *',
            { table: 'envelopes', column: 'budget', amount: -req.transferBudget, id: req.envelopeFromId }
          );
          const to = await t.one(
            'UPDATE ${table:name} SET ${column:name} = ${column:name} + ${amount:csv} WHERE id = ${id:csv} RETURNING *',
            { table: 'envelopes', column: 'budget', amount: req.transferBudget, id: req.envelopeToId }
          );
          return { from, to };
        });
        req.updatedEnvelopes = result;
        next();
      } catch (err) {
        // ✅ 所有数据库错误(连接失败、SQL 错误、约束冲突等)均由 db.tx 自动捕获并回滚
        next(err);
      }
    }

?️ 进阶优化:复用查询逻辑(推荐)

为保持代码可维护性,可将参数化查询封装为纯函数,但务必接受 t 参数并默认回退到 db

// ✅ 安全的可复用查询函数:支持事务内(t)和独立(db)两种上下文
function addToColumn(tableName, columnName, entryId, amountToAdd, t = db) {
  return t.one(
    'UPDATE ${table:name} SET ${column:name} = ${column:name} + ${amount:csv} WHERE id = ${id:csv} RETURNING *',
    { table: tableName, column: columnName, amount: amountToAdd, id: entryId }
  );
}

// 在事务中调用(自动使用 t)
async function transferEnvelopeBudgetByIds(req, res, next) {
  try {
    const result = await db.tx(async t => {
      const [from, to] = await Promise.all([
        addToColumn('envelopes', 'budget', req.envelopeFromId, -req.transferBudget, t),
        addToColumn('envelopes', 'budget', req.envelopeToId, req.transferBudget, t)
      ]);
      return { from, to };
    });
    req.updatedEnvelopes = result;
    next();
  } catch (err) {
    next(err);
  }
}
? 提示:Promise.all() 在此场景下是安全的,因为所有 Promise 均由 t 创建,其 rejection 会被 db.tx() 统一捕获并触发自动回滚。若需严格顺序执行(如依赖前序结果),则改用 await 逐个调用。

⚠️ 关键注意事项

  • t.batch() 已废弃:pg-promise v10+ 中 Task.batch() 方法已被移除,官方文档明确推荐直接使用 Promise.all() 或 await 序列。
  • 绝不提前执行 Promise:任何数据库操作(如 db.one(), t.one())都应在 db.tx() 回调内部调用,确保其生命周期完全处于事务控制之下。
  • 错误处理集中化:事务内的所有错误应由 db.tx() 自动处理(自动回滚),外部只需 try/catch 捕获最终异常并交由 Express 错误中间件处理,避免在每个查询后 .catch()(这会中断事务链路)。
  • 环境健壮性:建议全局监听 process.on('unhandledRejection') 作为兜底,防止遗漏的 Promise 错误导致进程退出:
    process.on('unhandledRejection', (reason, promise) => {
      console.error('Unhandled Rejection at:', promise, 'reason:', reason);
      // 记录日志、告警,但不要在此处调用 process.exit()
    });

遵循以上模式,即可彻底规避“Uncaught Exception”,构建出高可靠性、易维护的 PostgreSQL 事务逻辑。

相关专题

更多
什么是中间件
什么是中间件

中间件是一种软件组件,充当不兼容组件之间的桥梁,提供额外服务,例如集成异构系统、提供常用服务、提高应用程序性能,以及简化应用程序开发。想了解更多中间件的相关内容,可以阅读本专题下面的文章。

175

2024.05.11

Golang 中间件开发与微服务架构
Golang 中间件开发与微服务架构

本专题系统讲解 Golang 在微服务架构中的中间件开发,包括日志处理、限流与熔断、认证与授权、服务监控、API 网关设计等常见中间件功能的实现。通过实战项目,帮助开发者理解如何使用 Go 编写高效、可扩展的中间件组件,并在微服务环境中进行灵活部署与管理。

212

2025.12.18

js正则表达式
js正则表达式

php中文网为大家提供各种js正则表达式语法大全以及各种js正则表达式使用的方法,还有更多js正则表达式的相关文章、相关下载、相关课程,供大家免费下载体验。

510

2023.06.20

js获取当前时间
js获取当前时间

JS全称JavaScript,是一种具有函数优先的轻量级,解释型或即时编译型的编程语言;它是一种属于网络的高级脚本语言,主要用于Web,常用来为网页添加各式各样的动态功能。js怎么获取当前时间呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

244

2023.07.28

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

253

2023.08.03

js是什么意思
js是什么意思

JS是JavaScript的缩写,它是一种广泛应用于网页开发的脚本语言。JavaScript是一种解释性的、基于对象和事件驱动的编程语言,通常用于为网页增加交互性和动态性。它可以在网页上实现复杂的功能和效果,如表单验证、页面元素操作、动画效果、数据交互等。

5244

2023.08.17

js删除节点的方法
js删除节点的方法

js删除节点的方法有:1、removeChild()方法,用于从父节点中移除指定的子节点,它需要两个参数,第一个参数是要删除的子节点,第二个参数是父节点;2、parentNode.removeChild()方法,可以直接通过父节点调用来删除子节点;3、remove()方法,可以直接删除节点,而无需指定父节点;4、innerHTML属性,用于删除节点的内容。

471

2023.09.01

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

206

2023.09.04

漫蛙2入口地址合集
漫蛙2入口地址合集

本专题整合了漫蛙2入口汇总,阅读专题下面的文章了解更多详细内容。

162

2026.01.06

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 8.1万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号