Oracle 若存在则更新update ,不存在则插入insert

本文介绍了在数据库操作中如何使用MERGE INTO语句来高效地执行插入或更新操作,避免了先查询再更新的传统方式。通过示例展示了如何在存在匹配时更新,不存在时插入数据。同时提到了使用该语法时需要注意的事项,如using语句的正确使用,避免在on条件中使用更新列等。

今天开发碰到了一个很常见的问题,就是需要根据数据库中是否存在来insert或update,以前的做法是
1.查询记录是否存在

select count(1) from table_name where ****;
2.判断
如果>=1则update table_name
如果==0则insert table_name
新学习到的方法是使用MERGE INTO语法

MERGE INTO 表名
USING 表名/视图/子查询 ON 连接条件
– 当匹配得上连接条件时
WHEN MATCHED THEN
更新、删除操作
– 当匹配不上连接条件时
WHEN NOT MATCHED THEN
更新、删除、插入操作
MERGE INTO table_name alias1
USING (table | view | sub_query) alias2
ON (join condition)
WHEN MATCHED THEN
UPDATE table_name SET col1 = col_val1, col2 = col2_val
WHEN NOT MATCHED THEN
INSERT (column_list) VALUES (column_values);
简单翻译一下
操作alias1表中的数据
通过ON 中的条件进行过滤
存在执行更新操作
不存在执行插入操作

假设studen中name字段唯一
现在的业务场景是student中有叫小明的则更新小明的信息(age)
没有叫小明的则插入小明的信息(name,age)
实现如下

merge into student stu
using (SELECT
nvl(max(student .name),0) name
FROM student
WHERE student .name= :‘小明’) s
on (stu.name=s.name)
when matched then
update set stu.age = ‘11’
when not matched then
insert ( name, age) values ( ‘小明’, ‘10’);
需要注意的地方
1.如果using中的语句什么都查不出来,那么insert是不会执行的
2.on中的条件不能作为更新列,否则会报错
3.on中的条件一定要过滤准确,否则会整表更新

目前就理解这么多

2020年4月27日记录
发现一个问题,merge into 用as别名时会报错,直接空格别名没问题

4人点赞
开发问题

作者:在简单一点
链接:https://www.jianshu.com/p/a6efb0ec640d
来源:简书
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

<think>好的,我需要帮助用户解决在SQL中实现“插入前判断记录是否存在存在更新”的问题。首先,我应该回顾用户提供的引用内容,看看有没有相关的信息。 用户引用了三个资料:第一个提到INSERT的四种类型,包括有条件INSERT ALL和FIRST,这可能涉及到条件插入。第二个引用是关于基本的INSERT INTO语句,其中提到自动更新的id字段,但没有直接提到存在性检查。第三个引用提到数据库管理系统,包括SQL Server和非关系型数据库,但可能直接相关。 用户的需求是“upsert”(更新插入),这在SQL中通常通过特定语法实现。同的数据库系统可能有同的方法,比如MySQL的ON DUPLICATE KEY UPDATE,PostgreSQL的INSERT ... ON CONFLICT,以及SQLite的REPLACE INTO。此外,MERGE语句在SQL标准中也是常用的,但需要确认数据库是否支持。 需要指出用户可能没有说明使用的数据库类型,因此需要分情况讨论。同时,可以建议使用事务或存储过程来确保原子性,避免竞态条件。另外,引用中的有条件INSERT可能是指INSERT ALL或FIRST,但可能够直接解决upsert问题,需要进一步解释。 总结回答结构:先介绍upsert概念,然后分数据库类型说明同实现方式,最后给出通用方法或注意事项。需要确保语法正确,使用中文,并引用提供的资料中的相关内容,例如提到有条件INSERT类型时引用[^1],或者基本INSERT语句引用[^2]。</think>在SQL中实现“插入前检查记录是否存在存在更新”的功能通常称为**upsert**(update + insert)。根据同的数据库系统,实现方式有所差异。以下是几种常见方法及步骤说明: --- ### 方法1:使用数据库特定的语法 #### MySQL 使用`ON DUPLICATE KEY UPDATE`: ```sql INSERT INTO table (id, name, value) VALUES (1, 'test', 100) ON DUPLICATE KEY UPDATE name = 'test', value = 100; ``` **条件**:必须存在唯一索引或主键约束。当插入导致主键或唯一索引冲突时,自动执行更新操作[^2]。 #### PostgreSQL 使用`INSERT ... ON CONFLICT`: ```sql INSERT INTO table (id, name, value) VALUES (1, 'test', 100) ON CONFLICT (id) DO UPDATE SET name = EXCLUDED.name, value = EXCLUDED.value; ``` **条件**:需明确指定冲突的列(如`id`)。 #### SQLite 使用`REPLACE INTO`(先删除旧记录再插入): ```sql REPLACE INTO table (id, name, value) VALUES (1, 'test', 100); ``` --- ### 方法2:通用SQL标准(MERGE语句) 适用支持`MERGE`的数据库(如Oracle、SQL Server): ```sql MERGE INTO table AS target USING (SELECT 1 AS id, 'test' AS name, 100 AS value) AS source ON (target.id = source.id) WHEN MATCHED THEN UPDATE SET name = source.name, value = source.value WHEN NOT MATCHED THEN INSERT (id, name, value) VALUES (source.id, source.name, source.value); ``` **特点**:通过条件匹配实现原子性操作[^1]。 --- ### 方法3:事务 + 条件判断(通用但非原子性) 1. **先查询是否存在**: ```sql SELECT id FROM table WHERE id = 1; ``` 2. **根据查询结果执行操作**: ```sql -- 如果存在更新 UPDATE table SET name = 'test', value = 100 WHERE id = 1; -- 如果存在插入 INSERT INTO table (id, name, value) VALUES (1, 'test', 100); ``` **缺点**:需手动处理事务以确保原子性,否则可能因并发操作导致数据一致。 --- ### 注意事项 1. **原子性**:优先使用数据库原生支持的upsert语法(如`ON DUPLICATE KEY UPDATE`或`MERGE`),避免竞态条件。 2. **性能**:批量操作时,原生语法通常比“先查后插”更高效。 3. **索引要求**:部分语法依赖唯一索引或主键约束,需提前定义。 ---
评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值