如何在Postgres的INSERT ON CONFLICT DO UPDATE中返回列的旧值?

8
我会尝试在Postgres中运行“upsert”,类似于:
INSERT INTO my_table (
    id, -- unique key
    name,
    hash
) VALUES (
    '4b544dea-b355-463c-8fba-40c36ac7cb0c',
    'example',
    '0x0123456789'
) ON CONFLICT (
    id
) DO UPDATE SET
    name = 'example',
    hash = '0x0123456789'
RETURNING
    OLD.hash;

我希望能够返回hash列的先前值(上面的示例由于OLD不是有效的表别名而无效)。重要的是,我正在尝试找到一种在负载下不会引起冲突的方法。这是否可能?还是唯一的解决方案是在事务中进行读取操作后再进行写入操作?

2个回答

6
如果您愿意更新架构,另一个可能的解决方案是将先前的值存储在另一列中:
ALTER table my_table add column prev_hash text;

INSERT INTO my_table (
    id, -- unique key
    hash
) VALUES (
    1,
    'a'
) ON CONFLICT (
    id
) DO UPDATE SET
    hash = excluded.hash,
    prev_hash = my_table.hash
RETURNING
    id,
    hash,      -- will be the new hash
    prev_hash  -- will be the old hash

4
我最终找到了一个解决方法,虽然它并不能严格保证原子性,但总的来说,根据您的使用情况,这可能是一个不错的策略。
INSERT INTO my_table (
    id, -- unique key
    name,
    hash
) VALUES (
    '4b544dea-b355-463c-8fba-40c36ac7cb0c',
    'example',
    '0x0123456789'
) ON CONFLICT (
    id
) DO UPDATE SET
    name = 'example',
    hash = '0x0123456789'
RETURNING
    name,
    hash, -- will be the new hash
    (SELECT hash FROM my_table WHERE my_table.id = '4b544dea-b355-463c-8fba-40c36ac7cb0c') -- will be the old hash
    ;

如果存在插入操作(即未触发ON CONFLICT),则RETURNING块将失败,因为记录以前不存在。 - tedtanner

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接