执行 SELECT 或 INSERT

如何解决执行 SELECT 或 INSERT

我想在 PostgreSQL 中编写类似 CASE 语句的内容,以便我想从表中 SELECT,如果没有找到,我想 INSERT 进入代替桌子。

我的示例表 allocated_ideas 如下所示:

challenge_id | user_id    | idea_id        | round | is_completed
-------------+------------+----------------+-------+-------------
 '5d956daa'  | '9afbca19' | '3798826ae522' |   5   | false

我想在 SQL 函数中做这样的事情:

SELECT EXISTS (
SELECT 1 from allocated_ideas ai
where ai.user_id = '9afbca19' 
and ai.challenge_id = '5d956daa'
    and ai.is_completed is false

如果此条件为真,则返回 idea_id。否则,将值插入表中。伪代码:

if (above query is true) {
    SELECT idea_id from allocated_ideas ai 
    where ai.user_id = '9afbca19' 
    and ai.challenge_id = '5d956daa'
        and ai.is_completed is false 
} else {
  INSERT VALUES INTO allocated_ideas('67879','46578','978798',6,false)
}

解决方法

表上没有并发写入负载,您可以简单地:

WITH sel AS (
   SELECT idea_id
   FROM   allocated_ideas
   WHERE  user_id = '9afbca19' 
   AND    challenge_id = '5d956daa'
   AND    is_completed IS false 
   ),ins AS (
                             -- spell out target columns!
   INSERT INTO allocated_ideas (challenge_id,user_id,idea_id,round,is_completed)
   SELECT '67879','46578','978798',6,false
   WHERE  NOT EXISTS (TABLE sel)               -- only when sel is empty!
   -- RETURNING idea_id                        -- unused
)
TABLE sel;

数据修改 CTE ins 总是执行到完成,即使它在最终的主命令中没有被引用。 (一个简单的 SELECT 在没有被引用的情况下不会被执行。)见:

如果您省略 INSERT 的目标列,它可能会无声地(或嘈杂地)中断!

您没有指定从 idea_id 返回新的 INSERT。如果需要,您可能还希望能够区分案例。我为此添加了 op 列:

WITH sel AS (
   -- like above,ins AS (
   -- like above
   RETURNING idea_id                        -- used now
   )
SELECT 'sel' AS op,* FROM sel
UNION ALL
SELECT 'ins' AS op,* FROM ins;

查询只能返回单行,并且 INSERT 在任何情况下都会执行(如果 sel 找到一行,则不执行任何操作),因此我们不需要 LIMIT 1

通常情况下,可能存在并发写入负载,但您希望避免竞争条件。见:

,

这是使用通用表表达式 (CTE) 的一种方法:

WITH sel AS (
    SELECT challenge_id,is_completed
    FROM allocated_ideas
    WHERE user_id = '9afbca19' AND challenge_id = '5d956daa' AND is_completed = FALSE
),ins as (
    INSERT INTO allocated_ideas (challenge_id,is_completed)
    SELECT '67879',FALSE
    WHERE NOT EXISTS (TABLE sel)
    RETURNING *
)
SELECT * FROM sel
UNION ALL
SELECT * FROM ins

请注意,这仅在没有修改表的并发事务时有效。如果您希望并发写入,请先尝试使用 ON CONFLICT 的 INSERT。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其他元素将获得点击?
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。)
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbcDriver发生异常。为什么?
这是用Java进行XML解析的最佳库。
Java的PriorityQueue的内置迭代器不会以任何特定顺序遍历数据结构。为什么?
如何在Java中聆听按键时移动图像。
Java“Program to an interface”。这是什么意思?
Java在半透明框架/面板/组件上重新绘画。
Java“ Class.forName()”和“ Class.forName()。newInstance()”之间有什么区别?
在此环境中不提供编译器。也许是在JRE而不是JDK上运行?
Java用相同的方法在一个类中实现两个接口。哪种接口方法被覆盖?
Java 什么是Runtime.getRuntime()。totalMemory()和freeMemory()?
java.library.path中的java.lang.UnsatisfiedLinkError否*****。dll
JavaFX“位置是必需的。” 即使在同一包装中
Java 导入两个具有相同名称的类。怎么处理?
Java 是否应该在HttpServletResponse.getOutputStream()/。getWriter()上调用.close()?
Java RegEx元字符(。)和普通点?