0

0

SQL UPSERT / ON DUPLICATE KEY UPDATE / ON CONFLICT 的写法与性能对比

冷漠man

冷漠man

发布时间:2026-02-15 17:50:03

|

433人浏览过

|

来源于php中文网

原创

mysql的on duplicate key update要求主键或唯一索引存在,否则报错;postgresql的on conflict必须显式指定冲突字段;sqlite的replace into是先删后插,可能引发级联删除。

sql upsert / on duplicate key update / on conflict 的写法与性能对比

MySQL 用 ON DUPLICATE KEY UPDATE 做 UPSERT,但主键或唯一索引必须存在

没定义 UNIQUEPRIMARY KEYON DUPLICATE KEY UPDATE 直接报错:ERROR 1062 (23000): Duplicate entry '...' for key 'PRIMARY' —— 这个错误不是“发生了冲突”,而是“连判断冲突的依据都没有”。

常见误操作是只建了普通索引,或者忘了给业务字段加 UNIQUE。比如想根据 email 去重更新,却只加了 INDEX email_idx (email),没加 UNIQUE

  • INSERT INTO users (email, name) VALUES ('a@b.com', 'Alice') ON DUPLICATE KEY UPDATE name = VALUES(name) 要求 email 列有 UNIQUE 约束(可以是单独列,也可以是联合唯一)
  • 如果冲突字段是联合唯一(如 UNIQUE (tenant_id, external_id)),那 VALUES() 里必须包含这两个字段,否则无法触发更新
  • 注意 VALUES(col) 是指本次 INSERT 语句中该列的值,不是表里原来的值;别写成 name = name,那会清空字段

PostgreSQL 的 ON CONFLICT 必须显式指定冲突目标,不能只靠索引名猜

PostgreSQL 不会自动识别“哪个唯一约束被违反”,你得明确告诉它:是按主键?还是某个唯一索引?还是某几个字段的组合?漏写或写错就直接报错:ON CONFLICT clause does not match any unique constraint

典型场景是想按 email 去重,但表里有多个唯一约束(比如 PRIMARY KEY + UNIQUE (email)),不指定就失败。

  • 正确写法:INSERT INTO users (email, name) VALUES ('a@b.com', 'Alice') ON CONFLICT (email) DO UPDATE SET name = EXCLUDED.name
  • 如果冲突目标是联合唯一索引 idx_tenant_ext (tenant_id, external_id),就得写 ON CONFLICT (tenant_id, external_id),不能写索引名
  • EXCLUDED 是个伪表,代表本次被拒绝插入的那行数据;别写成 NEWINSERTED,它们不存在
  • 如果字段允许 NULL,而冲突条件含 NULL,注意 PostgreSQL 中 NULL = NULLFALSE,所以 ON CONFLICT (nullable_col)NULL 值不生效

SQLite 的 REPLACE INTO 是删除+插入,可能意外删掉关联数据

REPLACE INTO 看似最简单,但它底层是先 DELETEINSERT。一旦表上有外键级联(比如 ON DELETE CASCADE),或者触发器监听了 DELETE,行为就不可控了。

Dream Machine
Dream Machine

Dream Machine 是由 Luma AI 开发的一款 AI 视频生成工具,可以快速将文本和图像转换为高质量的视频内容。

下载

比如用户表和地址表有外键,用 REPLACE INTO users 更新邮箱,地址记录可能被连带删掉 —— 这不是“更新”,是“先杀后生”。

  • 真正安全的 SQLite UPSERT 是 INSERT OR REPLACE INTO(等价于 REPLACE INTO),但同样走 delete-insert 流程
  • 若需保留原行 ID 或避免级联删除,得用 INSERT OR IGNORE + 单独 UPDATE 两步,靠事务包住
  • 没有 VALUES()EXCLUDED 这类语法,所有更新值都得手写,容易漏字段或写错变量名

跨数据库写法不通用,性能差异集中在索引扫描和锁粒度

三者看着都是“插入或更新”,但执行路径完全不同:MySQL 在唯一索引上做一次查找+条件更新;PostgreSQL 先尝试插入,冲突时回退并执行 DO UPDATE;SQLite 删除再插,IO 和锁开销更大。

高并发下最容易出问题的是 MySQL 的 ON DUPLICATE KEY UPDATE:它会对冲突的索引记录加 next-key lock,可能比预期锁得更宽,引发间隙锁等待。

  • PostgreSQL 的 ON CONFLICT 默认只锁冲突行,但如果 DO UPDATE 涉及其他索引字段,可能触发额外索引维护锁
  • SQLite 在 WAL 模式下 REPLACE 仍会阻塞读,因为 delete 和 insert 是两个独立操作,中间有可见窗口
  • 别指望用 ORM 的“upsert”抽象来屏蔽差异 —— Django 的 update_or_create、SQLAlchemy 的 insert().on_conflict_do_update() 底层仍是各自方言,参数含义和边界行为不一致

最常被忽略的一点:所有这些语法都要求冲突判断字段有高效索引。没建索引的 ON CONFLICT (unindexed_col) 会全表扫,还报错;MySQL 里没索引的 ON DUPLICATE KEY 根本不工作。索引不是可选优化项,是功能前提。

数码产品性能查询
数码产品性能查询

该软件包括了市面上所有手机CPU,手机跑分情况,电脑CPU,电脑产品信息等等,方便需要大家查阅数码产品最新情况,了解产品特性,能够进行对比选择最具性价比的商品。

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
数据分析工具有哪些
数据分析工具有哪些

数据分析工具有Excel、SQL、Python、R、Tableau、Power BI、SAS、SPSS和MATLAB等。详细介绍:1、Excel,具有强大的计算和数据处理功能;2、SQL,可以进行数据查询、过滤、排序、聚合等操作;3、Python,拥有丰富的数据分析库;4、R,拥有丰富的统计分析库和图形库;5、Tableau,提供了直观易用的用户界面等等。

943

2023.10.12

SQL中distinct的用法
SQL中distinct的用法

SQL中distinct的语法是“SELECT DISTINCT column1, column2,...,FROM table_name;”。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

334

2023.10.27

SQL中months_between使用方法
SQL中months_between使用方法

在SQL中,MONTHS_BETWEEN 是一个常见的函数,用于计算两个日期之间的月份差。想了解更多SQL的相关内容,可以阅读本专题下面的文章。

375

2024.02.23

SQL出现5120错误解决方法
SQL出现5120错误解决方法

SQL Server错误5120是由于没有足够的权限来访问或操作指定的数据库或文件引起的。想了解更多sql错误的相关内容,可以阅读本专题下面的文章。

1656

2024.03.06

sql procedure语法错误解决方法
sql procedure语法错误解决方法

sql procedure语法错误解决办法:1、仔细检查错误消息;2、检查语法规则;3、检查括号和引号;4、检查变量和参数;5、检查关键字和函数;6、逐步调试;7、参考文档和示例。想了解更多语法错误的相关内容,可以阅读本专题下面的文章。

369

2024.03.06

oracle数据库运行sql方法
oracle数据库运行sql方法

运行sql步骤包括:打开sql plus工具并连接到数据库。在提示符下输入sql语句。按enter键运行该语句。查看结果,错误消息或退出sql plus。想了解更多oracle数据库的相关内容,可以阅读本专题下面的文章。

1231

2024.04.07

sql中where的含义
sql中where的含义

sql中where子句用于从表中过滤数据,它基于指定条件选择特定的行。想了解更多where的相关内容,可以阅读本专题下面的文章。

583

2024.04.29

sql中删除表的语句是什么
sql中删除表的语句是什么

sql中用于删除表的语句是drop table。语法为drop table table_name;该语句将永久删除指定表的表和数据。想了解更多sql的相关内容,可以阅读本专题下面的文章。

434

2024.04.29

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

139

2026.02.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号