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

Bur*_*urg 5 sql postgresql

我正在尝试在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;
Run Code Online (Sandbox Code Playgroud)

我想返回该hash列的先前值(由于OLD不是有效的表别名,以上示例不是有效的查询)。重要的是,我正在尝试找到一种在负载下不会引起冲突的方法。这有可能吗?还是在事务中执行先写后读的唯一解决方案?

Sco*_*t B 7

如果您愿意更新架构,另一个可能的答案是将以前的值存储在另一列中:

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
Run Code Online (Sandbox Code Playgroud)


Bur*_*urg 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
    ;
Run Code Online (Sandbox Code Playgroud)