node.js - 在 postgres 中删除和插入的竞争条件
问题描述
我正在开发一个节点项目,我在其中导入了 pg 库以进行数据库操作。我有一个 Kafka 队列,我从中获取事件并将它们存储在数据库中。我正在从 kafka 获取订单,每次更新订单时都会生成一个新事件,我需要删除旧的订单详细信息并用新的替换它们。
下面是代码
async function saveOrders(orders: Array<TransactionOnOrders>) {
const client = await pool.connect()
try {
await client.query('BEGIN')
if (orders.length) {
const deleted = await deleteOrders(client, orders[0].orderId)
logger.debug(`deleted rowCount ${deleted.rowCount} ${orders[0].orderId}`)
}
const queries = orders.map(ord => saveTransactionOnOrders(client, ord))
await Promise.all(queries)
await client.query('COMMIT')
} catch (e) {
await client.query('ROLLBACK')
throw e
} finally {
client.release()
}
}
订单更新非常频繁,我们收到大量事件,创建竞争条件导致记录不被删除并插入额外记录。例如:假设我们收到 order123 的事件并且事务正在进行中,直到它完成 order123 的另一个事件的时间,因此删除查询返回 0 行受影响,插入查询插入另一行导致 2 行应该只有一个记录存在。
我试图更改效果不佳并导致错误的隔离级别
await client.query('BEGIN TRANSACTION ISOLATION LEVEL REPEATABLE READ')
await client.query('BEGIN TRANSACTION ISOLATION LEVEL SERIALIZABLE')
我在这里做错了什么还是有更好的方法来处理上述情况?
解决方案
如果您更新行而不是删除它们并重新创建它们,这可能会更容易。在这种情况下,您可以依靠行锁来防止并发更新。
推荐阅读
- java - Spring Integration:仅在处理完上次轮询的所有结果后才启动 JPA 轮询
- javascript - 验证给定格式的字符串
- django - Django 基于唯一键的连接模型
- c++ - 不能编译归并排序树结构?
- python - Tkinter 不在窗口上打印
- javascript - Laravel-Mix 与 BrowserSync 可处理除 Vue.js 组件之外的所有内容
- php - 我的验证码不起作用。警告:imagettftext():在第 35 行的 W:\domains\shop\reg\reg_captcha.php 中找不到/打开字体
- r - 如果数据来自不同的数据集,如何将 ggplots 添加到一个中?以及如何将几何功能区添加到离散数据?
- macos - 使用 Mac 时如何导出代码?
- active-directory - 无法使用 Active Directory 凭证登录到 Amazon Linux 服务器