insertOnDuplicateKey 和 upsert 区别

二者是一样的

insertOnDuplicateKey 实际上是 ON DUPLICATE KEY UPDATE,是MySQL的特有写法

类似的有 ON CONFICT() ,这个是 pgsql 的类似写法

在数据库操作中,INSERT ON DUPLICATE KEY UPDATE(MySQL特有语法)和UPSERT(通用概念,不同数据库实现不同)都是用于处理**"如果记录存在则更新,不存在则插入"**的场景。以下是它们的详细对比和关键区别:


1. 核心概念

(1) INSERT ON DUPLICATE KEY UPDATE(MySQL)

  • 定义 :MySQL特有的SQL语法,当插入数据时违反唯一键(UNIQUE KEY)或主键约束时,自动执行更新操作。

  • 适用场景:仅适用于MySQL及其衍生数据库(如MariaDB)。

  • 示例

    复制代码

    sql

    复制代码
    `INSERT INTO orders (order_id, product_id, quantity, price)
    VALUES (1, 100, 5, 10.0)
    ON DUPLICATE KEY UPDATE quantity = quantity + 1, price = VALUES(price);`
    • 如果order_idproduct_id的组合已存在,则更新quantityprice
    • VALUES(price)表示引用插入时的值。

(2) UPSERT(通用概念)

  • 定义 :数据库操作的通用术语,表示"插入或更新"(U pdate or Insert),是SQL标准中定义的原子操作。
  • 不同数据库的实现
    • PostgreSQLINSERT ... ON CONFLICT DO UPDATE
    • SQLiteINSERT OR REPLACEON CONFLICT
    • OracleMERGE语句
    • SQL ServerMERGE语句
    • JPA/Hibernate :通过@Entity@DynamicUpdatesaveOrUpdate方法模拟。

2. 关键区别

对比项 INSERT ON DUPLICATE KEY UPDATE(MySQL) UPSERT(通用实现)
语法来源 MySQL特有 SQL标准概念,不同数据库实现不同
冲突条件 仅基于唯一键或主键冲突 可基于唯一约束、主键或其他条件(如PostgreSQL的ON CONFLICT (column)
更新范围 只能更新冲突行的字段 可灵活控制更新哪些字段(如PostgreSQL的DO UPDATE SET
多行冲突处理 逐行处理冲突,每行独立判断 某些数据库(如PostgreSQL)支持批量冲突处理
返回值 返回受影响的行数(插入或更新) 返回值因数据库而异(如PostgreSQL返回特殊状态)
跨数据库兼容性 仅MySQL/MariaDB支持 需根据数据库调整语法(如PostgreSQL/Oracle)

3. 具体数据库实现对比

(1) MySQL

复制代码

sql

复制代码
`-- 插入或更新(基于唯一键冲突)
INSERT INTO users (id, name, email) 
VALUES (1, 'Alice', 'alice@example.com')
ON DUPLICATE KEY UPDATE name = VALUES(name), email = VALUES(email);`
  • 特点:简单直接,但仅限MySQL。

(2) PostgreSQL

复制代码

sql

复制代码
`-- 插入或更新(基于唯一约束冲突)
INSERT INTO users (id, name, email) 
VALUES (1, 'Alice', 'alice@example.com')
ON CONFLICT (id) DO UPDATE 
SET name = EXCLUDED.name, email = EXCLUDED.email;`
  • 特点
    • ON CONFLICT (column)指定冲突条件。
    • EXCLUDED表引用插入时的值。

(3) SQLite

复制代码

sql

复制代码
`-- 方式1:替换整行(危险!可能丢失数据)
INSERT OR REPLACE INTO users (id, name, email) 
VALUES (1, 'Alice', 'alice@example.com');

-- 方式2:条件更新(更安全)
INSERT INTO users (id, name, email) 
VALUES (1, 'Alice', 'alice@example.com')
ON CONFLICT(id) DO UPDATE SET name = excluded.name;`
  • 注意OR REPLACE会删除旧行再插入新行,可能导致外键约束问题。

(4) Oracle/SQL Server

复制代码

sql

复制代码
`-- Oracle/SQL Server使用MERGE语句
MERGE INTO users target
USING (SELECT 1 AS id, 'Alice' AS name, 'alice@example.com' AS email FROM dual) source
ON (target.id = source.id)
WHEN MATCHED THEN
    UPDATE SET target.name = source.name, target.email = source.email
WHEN NOT MATCHED THEN
    INSERT (id, name, email) VALUES (source.id, source.name, source.email);`
  • 特点:功能强大,但语法复杂。

4. 如何选择?

  • MySQL项目 :直接用INSERT ON DUPLICATE KEY UPDATE,简单高效。
  • 跨数据库项目
    • 使用JPA/Hibernate的saveOrUpdate(需实体类配置正确)。
    • 或根据数据库类型动态生成SQL(如PostgreSQL用ON CONFLICT,Oracle用MERGE)。
  • 高并发场景 :注意锁竞争问题(如MySQL的INSERT ... ON DUPLICATE KEY可能引发间隙锁)。

5. 代码示例(Spring Data JPA)

如果使用Spring Data JPA,可以通过以下方式模拟UPSERT:

复制代码

java

复制代码
`@Repository
public interface UserRepository extends JpaRepository<User, Long> {
    @Modifying
    @Query(value = "INSERT INTO users (id, name, email) " +
                   "VALUES (:id, :name, :email) " +
                   "ON CONFLICT (id) DO UPDATE SET name = :name, email = :email", 
           nativeQuery = true)
    void upsertUser(@Param("id") Long id, @Param("name") String name, @Param("email") String email);
}`
  • 注意 :需启用@Modifying和事务(@Transactional)。

总结

  • INSERT ON DUPLICATE KEY UPDATE:MySQL专属,简单直接。
  • UPSERT :通用概念,需根据数据库选择具体实现(如PostgreSQL的ON CONFLICT、Oracle的MERGE)。
  • 跨数据库兼容性:优先考虑JPA/Hibernate或抽象层,避免硬编码SQL。
相关推荐
nbsaas-boot2 小时前
SQL Server 存储过程设计规范(事务与异常处理)
linux·数据库·设计规范
大学生资源网2 小时前
基于springboot的万亩助农网站的设计与实现源代码(源码+文档)
java·spring boot·后端·mysql·毕业设计·源码
q_19132846952 小时前
基于SpringBoot2+Vue2的诗词文化传播平台
vue.js·spring boot·mysql·程序员·计算机毕业设计
为什么不问问神奇的海螺呢丶3 小时前
服务器巡检报告-基于categraf 采集数据-存入Prometheus-写入mysql后生成报告
服务器·mysql·prometheus
代码or搬砖3 小时前
SQL核心语法总结:从基础操作到高级窗口函数
java·数据库·sql
3 小时前
TIDB——TIKV——读写与coprocessor
数据库·分布式·tidb·
秋氘渔3 小时前
智演沙盘 —— 基于大模型的智能面试评估系统
python·mysql·django·drf
大猫和小黄4 小时前
若依微服务全面适配PostgreSQL-OpenGauss数据库
数据库·微服务·postgresql·若依
老徐电商数据笔记4 小时前
技术复盘第二篇:电商数据主题域划分企业级实践
大数据·数据库·数据仓库·零售·教育电商·技术面试