Oracle:merge into用法

文章目录

merge into

MERGE 是 Oracle9i 新增的语法,根据源表对目标表进行匹配查询,匹配成功时更新,不成功时插入

比单独的 update + insert 的方式效率要更高,尤其是on条件下有唯一索引的时候,效率更高。

使用场景

在写数据同步的脚本时,常常会遇到这样的需求:存在时更新,不存在时插入

merge into语法

sql 复制代码
MERGE INTO [target-table] T  --目标表 可以用别名
USING [source-table] S  --数据源表 可以是表、视图、子查询
ON([conditional expression] )  --关联条件
WHEN MATCHED THEN --当关联条件成立时 更新、删除,插入的where部分为可选 
UPDATE [target-table] SET T.column = S.column WHERE 限制条件
DELETE [target-table] WHERE 限制条件
WHEN NOT MATCHED THEN --当关联条件不成立时   
INSERT (column,...) VALUES('',...)

判断源表 S 和目标表 T 是否满足 ON 中的条件,如果满足则用 S 表去更新 T 表,如果不满足,则将 S 表数据插入 T 表中。但是有很多可选项,如下:

  • 普通模式
  • 只 update 或者只 insert
  • 无条件 insert 实现
  • 带 delete 的 update

测试表

sql 复制代码
-- 目标表
CREATE TABLE target ( 
	ID NUMBER NOT NULL, 
	NAME VARCHAR2 ( 12 ) NOT NULL, 
	YEAR NUMBER 
);
-- 源表
CREATE TABLE source (
	ID NUMBER NOT NULL,
	AID NUMBER NOT NULL,
	NAME VARCHAR2 ( 12 ) NOT NULL,
	YEAR NUMBER,
	CITY VARCHAR2 ( 12 ) 
);
-- 插入测试数据
INSERT INTO target
VALUES
	( 1, 'liuwei', 20 );
INSERT INTO target
VALUES
	( 2, 'zhangbin', 21 );
INSERT INTO target
VALUES
	( 3, 'fuguo', 20 );
INSERT INTO source
VALUES
	( 1, 2, 'zhangbin', 30, '吉林' );
INSERT INTO source
VALUES
	( 2, 4, 'yihe', 33, '黑龙江' );
INSERT INTO source
VALUES
	( 3, 3, 'fuguo', '', '山东' );

普通模式

merge使用注意点

1、如果using中的语句查询不出来数据,是不会执行insert方法的,因为这个语法是根据using 中的查询数据进行判断

sql 复制代码
merge into student a
using (select id  from student where id = '7') s
on (a.id = s.id )
when matched then
  update set a.student_name = '小明二号'
when not matched then
  insert (id, student_name, fk_class) values ('7', '小明', '2')

2、on 中的条件记得过滤准确,不然可能会执行全表更新

sql 复制代码
merge into student a
using (select count(1)cot,id  from student group by id ) s
on (a.id = s.id and cot > 0)
when matched then
  update set a.student_name = '小明二号'
when not matched then
  insert (id, student_name, fk_class) values ('7', '小明', '2')

这么写的话可以看出明显的错误,只要是id相等且cot大于0,那么查询出的都大于0,会执行全表更新

3、on 中的条件不能是更新操作列,不然会报错:ora-38104

sql 复制代码
merge into student a
using (select '7' as id from dual) s
on (a.id = s.id)
when matched then
  update set a.id = '7'
when not matched then
  insert (id, student_name, fk_class) values ('7', '小明', '2');

参考:

https://blog.csdn.net/weixin_44657888/article/details/124591434

https://www.jianshu.com/p/8f51ce60d9ba

相关推荐
悟能不能悟3 小时前
redis的红锁
数据库·redis·缓存
安当加密5 小时前
MySQL数据库透明加密(TDE)解决方案:基于国密SM4的合规与性能优化实践
数据库·mysql·性能优化
JH30735 小时前
第七篇:Buffer Pool 与 InnoDB 其他组件的协作
java·数据库·mysql·oracle
板凳坐着晒太阳5 小时前
ClickHouse 配置优化与问题解决
数据库·clickhouse
数据库生产实战5 小时前
解析Oracle 19C中并行INSERT SELECT的工作原理
数据库·oracle
AAA修煤气灶刘哥6 小时前
服务器指标多到“洪水泛滥”?试试InfluxDB?
数据库·后端·面试
阿沁QWQ7 小时前
MySQL服务器配置与管理
服务器·数据库·mysql
程序新视界8 小时前
MySQL“索引失效”的隐形杀手:隐式类型转换,你了解多少?
数据库·mysql·dba
Logintern098 小时前
windows如何设置mongodb的副本集
数据库·windows·mongodb
RestCloud10 小时前
在制造业数字化转型浪潮中,数据已成为核心生产要素。然而,系统割裂、数据滞后、开发运维成本高等问题,却像顽固的 “数据枷锁”,阻碍着企业发展。ETLCloud与
数据库·postgresql