- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章PostgreSQL的upsert实例操作(insert on conflict do)由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
建表语句:
1
2
3
4
5
6
7
8
9
10
11
12
|
DROP
TABLE
IF EXISTS
"goods"
;
CREATE
TABLE
"goods"
(
"store_cd"
int4
NOT
NULL
,
"good_cd"
varchar
(50)
COLLATE
"pg_catalog"
.
"default"
NOT
NULL
,
"name"
varchar
(255)
COLLATE
"pg_catalog"
.
"default"
);
INSERT
INTO
"goods"
VALUES
(101,
'1'
,
'张三'
);
INSERT
INTO
"goods"
VALUES
(102,
'2'
,
'李四'
);
INSERT
INTO
"goods"
VALUES
(103,
'3'
,
'王五'
);
ALTER
TABLE
"goods"
ADD
CONSTRAINT
"pr_cd_key"
PRIMARY
KEY
(
"store_cd"
,
"good_cd"
);
|
表数据:
数据存在则更新数据,不存在则插入数据 。
1
2
3
4
5
6
7
|
INSERT
INTO
GOODS
VALUES
( 104,
'4'
,
'赵六'
)
ON
CONFLICT
ON
CONSTRAINT
pr_key_cd DO
UPDATE
SET
NAME
=
'更新'
WHERE
GOODS.STORE_CD =
'104'
AND
GOODS.GOOD_CD =
'4'
|
pr_key_cd为必须为唯一主键,也可以用下面写法(注意:必须保证筛选出数据唯一) 。
1
2
3
4
5
6
7
|
INSERT
INTO
GOODS
VALUES
( 104,
'4'
,
'赵六'
)
ON
CONFLICT ( STORE_CD, GOOD_CD ) DO
UPDATE
SET
NAME
=
'更新'
WHERE
GOODS.STORE_CD =
'104'
AND
GOODS.GOOD_CD =
'4'
|
上面的两种的写法,是先执行insert如果主键冲突则执行update,没有冲突就执行insert了。要是想先执行update语句呢?
update更新失败执行insert,更新成功则执行update.
1
2
3
|
WITH
TABLE1
AS
(
UPDATE
GOODS
SET
NAME
=
'更新'
WHERE
STORE_CD =
'104'
AND
GOOD_CD =
'4'
RETURNING * )
INSERT
INTO
GOODS
SELECT
104,
'4'
,
'赵六'
WHERE
NOT
EXISTS (
SELECT
1
FROM
TABLE1
WHERE
STORE_CD =
'104'
AND
GOOD_CD =
'4'
)
|
补充:Postgresql插入或更新操作upsert 。
幂等性的一个要求是多次操作的结果一致。对于update操作,多次直接的结果都是最后update的值,是满足需求的。但对于insert,如果已经插入,第二次会报错,duplicate error, 主键重复或者unique key duplicate。所以需要做一下处理.
最简单的就是,try-catch,当报错的时候,调用update去更新,或者策略更简单点,直接返回就行,不需要更新,以第一条为准.
PostgreSQL从9.5之后就提供了原子的upsert语法: 不存在则插入,发生冲突可以update.
Inert语法 。
官方文档: https://www.postgresql.org/docs/devel/sql-insert.html 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
[
WITH
[ RECURSIVE ] with_query [, ...] ]
INSERT
INTO
table_name [
AS
alias ] [ ( column_name [, ...] ) ]
[ OVERRIDING { SYSTEM |
USER
} VALUE ]
{
DEFAULT
VALUES
|
VALUES
( { expression |
DEFAULT
} [, ...] ) [, ...] | query }
[
ON
CONFLICT [ conflict_target ] conflict_action ]
[ RETURNING * | output_expression [ [
AS
] output_name ] [, ...] ]
where
conflict_target can be one
of
:
( { index_column_name | ( index_expression ) } [
COLLATE
collation ] [ opclass ] [, ...] ) [
WHERE
index_predicate ]
ON
CONSTRAINT
constraint_name
and
conflict_action
is
one
of
:
DO NOTHING
DO
UPDATE
SET
{ column_name = { expression |
DEFAULT
} |
( column_name [, ...] ) = [ ROW ] ( { expression |
DEFAULT
} [, ...] ) |
( column_name [, ...] ) = ( sub-
SELECT
)
} [, ...]
[
WHERE
condition ]
|
index_column_name 。
The name of a table_name column. Used to infer arbiter indexes. Follows CREATE INDEX format. SELECT privilege on index_column_name is required. 。
index_expression 。
Similar to index_column_name, but used to infer expressions on table_name columns appearing within index definitions (not simple columns). Follows CREATE INDEX format. SELECT privilege on any column appearing within index_expression is required. 。
使用示例 。
创建表 。
1
2
3
4
5
|
CREATE
TABLE
"test"
.
"upsert_test"
(
"id"
int4
NOT
NULL
,
"name"
varchar
(255)
COLLATE
"pg_catalog"
.
"default"
)
;
|
当主键id冲突时,更新其他字段 。
1
2
3
4
|
INSERT
INTO
test.upsert_test(id,
"name"
)
VALUES
(1,
'm'
),(2,
'n'
),(4,
'c'
)
ON
conflict(id) DO
UPDATE
SET
"name"
= excluded.
name
;
|
did 冲突的主键 。
EXCLUDED 代指要插入的记录 。
当主键或者unique key发生冲突时,什么都不做 。
1
2
3
|
INSERT
INTO
test.upsert_test(id,
"name"
)
VALUES
(1,
'm'
),(2,
'n'
),(4,
'c'
)
ON
conflict(id) DO NOTHING;
|
以上为个人经验,希望能给大家一个参考,也希望大家多多支持我。如有错误或未考虑完全的地方,望不吝赐教.
原文链接:https://yunan.blog.csdn.net/article/details/100897511 。
最后此篇关于PostgreSQL的upsert实例操作(insert on conflict do)的文章就讲到这里了,如果你想了解更多关于PostgreSQL的upsert实例操作(insert on conflict do)的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我正在开发一些用于计费的数据库项目(PHP/MySQL)。 每当创建新账单时,我想生成一个由年、周和增量编号组成的账单编号。我想用触发器来做到这一点。触发器将使用现有的账单编号来查找增量编号,或者从新
我有一个 MySQL 插入,我正在使用 RAND 生成随机 INT 值问题是它不会插入到数据库中,因为该列接受 TINYINT ,如何将输出转换为 TINYINT。代码示例如下: INSERT INT
如果我想从单个插入中保存主键 (mytable_id),我已完成以下操作: CREATE OR REPLACE FUNCTION myfunct(ownerid text) RETURNS void
为了简单起见,假设我有两个表 用户表(id,email) 用户日志表(id, date) 无论 id 被插入到 user 表中,相同的 id 也应该被插入到 user_log 表中,否则事务应该失败。
为了简单起见,假设我有两个表 用户表(id,email) 用户日志表(id, date) 无论 id 被插入到 user 表中,相同的 id 也应该被插入到 user_log 表中,否则事务应该失败。
我知道在触发器中 - 至少对于 SQL Server - 人们永远不应该假设插入的表只有一行,这意味着触发器中这样的 SQL 通常是不好的: select @UserID = ID from inse
我正在使用 bigquery 对象中的方法 tabledata().insertAll 更新行列表。执行后,返回显示没有错误。但是,我的表仍然继续,没有写入任何数据。 可能是权限问题。如果是这样,为什
这是一个扩展 F# Recursive Tree Validation 的问题,我昨天已经很好地回答了。 这个问题涉及在现有树中插入一个 child 。这是我想使用的更新类型: type Name
我有 2 个表:用户和照片(在 mysql 数据库中)。 在这里你可以看到两个表之间的关系 User Photos -------------
我试图同时在不同的表上插入两行。 子查询INSERT INTO的AUTO_INCRMENT或id的值(如果已经存在)应该写入主查询中。 目前我有这个(仅用 3 个值简化),但它不起作用。我想知道是否有
我有一个 900 万行的表,由于其庞大的规模,我正在努力处理所有这些数据。 我想做的是在不覆盖数据的情况下将 IMPORT 一个 CSV 添加到表中。 在我做这样的事情之前; INSERT if no
我正在写新闻并将其插入到我的数据库中,我在 3 年前构建了代码并且运行良好,但我不能再插入了,我不明白为什么: $insert=mysqli_query($co,"INSERT INTO articl
我正在尝试编写一个简单的 INSERT 语句来将新用户添加到数据库中,但它不起作用,这意味着,我尝试插入到表中的数据都没有被插入。几个小时以来,我一直在尝试解决此问题,但没有成功。我尝试编写插入语句的
所以我有这个表格: http://i.imgur.com/vZYssQy.png 现在 ID、First Name、Last Name、DOB、Address、Phone Number 和 Post
在控制台中运行查询(SELECT 语句)时,从数据库检索到的数据以表格格式显示在数据库控制台工具窗口的结果 Pane 中。 我已经搜索过 datagrip Help我只是想知道是否有任何方法可以用于为
每当使用触发器插入行时,我都试图将另一行插入表中,但收到以下错误消息: The target table 'EDDSDBO.Redaction' of the DML statement cannot
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 这个问题似乎与 help center 中定义的范围内的编程无关。 . 关闭 6 年前。 Improve
我有以下代码片段: $get_data = mysqli_query ($connect, "SELECT * FROM users WHERE username = '$username'");
情况:需要向 SQLite 数据库中插入大量数据。 问题:我们可以使用两个语句来插入数据 - data = [("111", "222", "333"), ("AAA", "BBB", "CCC"),
我的数据库中有一个表 Teacher: TABLE Teacher ( ID CHAR (7) NOT NULL , name
我是一名优秀的程序员,十分优秀!