关于ORACLE merge into 的两个常见错误

------- MERGE语法简介 语法如下: MERGE hint INTO schema . table t_alias USING schema . { table | view | subquery } t_alias ON (condition) WHEN MATCHED THEN merge_update_clause WHEN NOT MATCHED THEN merge_insert_clause;

--好处:是执行 同时有插入和更新操作时效率最高的脚本 讲解前建表:

CREATE TABLE TEST_111111  (ID NUMBER(18),  NAME VARCHAR2(255)  );  INSERT INTO TEST_111111  VALUES (1,'小红');  INSERT INTO TEST_111111  VALUES (2,'小红');  CREATE TABLE TEST_222222  AS  SELECT * FROM TEST_111111  WHERE ID = 1;

  Oracle10g中MERGE的完善 在Oracle10g以后,Oracle的MERGE发生了改变  UPDATE和INSERT动作可只出现其一  --可以只出现update   MERGE INTO TEST_111111 T1  USING TEST_222222 T2  ON (T1.ID = T2.ID)  WHEN MATCHED THEN    UPDATE SET T1.NAME = T2.NAME; --也可选择仅仅INSERT目标表而不做任何UPDATE动作  MERGE INTO TEST_111111 T1  USING TEST_222222 T2  ON (T1.ID = T2.ID)  WHEN NOT MATCHED THEN    INSERT VALUES (T2.ID, T2.NAME);  --而9i 版本的 则update 与 insert  都必须存在  MERGE INTO TEST_111111 T1  USING TEST_222222 T2  ON (T1.ID = T2.ID)  WHEN MATCHED THEN    UPDATE SET T1.NAME = T2.NAME  WHEN NOT MATCHED THEN    INSERT VALUES (T2.ID, T2.NAME);

-----------两种最常见的错误:

-PART1.ora-30926 :无法在源表中获得一组稳定的行

INSERT INTO TEST_111111  VALUES (1,'小红');

上面这条语句执行两次,插入两条相同的记录

INSERT INTO TEST_222222   SELECT * FROM TEST_111111  WHERE ID = 1;

MERGE INTO TEST_111111 T1  USING TEST_222222 T2 ON (T1.NAME = T2.NAME )  WHEN MATCHED THEN  UPDATE SET T1.ID = 521  WHEN NOT MATCHED THEN  INSERT VALUES (T2.ID,T2.NAME);

这时候就会报ORA-30926:无法再源表中获得一组稳定的行

原因 :T1 表为源表,意思是 在 ON(CONDITION) 这里在做CONDITION 判断的时候,匹配到的T1中的数据不止一条,所以CONDITION 这里建议 以主键为条件,这样就避免了匹配到多条数据的问题。

解决方案:知道了出错原因,解决起来就有方向可寻

假设 iD为主键,脚本改成

MERGE INTO TEST_111111 T1  USING TEST_222222 T2  ON (T1.ID= T2.ID)  WHEN MATCHED THEN  UPDATE SET T1.NAME = T2.NAME  WHEN NOT MATCHED THEN  INSERT VALUES (T2.ID,T2.NAME);

--PART2:ora-38104:无法更新on子句中引用的列

 MERGE INTO TEST_111111 T1  USING TEST_222222 T2  ON (T1.NAME = T2.NAME )  WHEN MATCHED THEN  UPDATE SET T1.NAME = T2.NAME  WHEN NOT MATCHED THEN  INSERT VALUES (T2.ID,T2.NAME);

出错原因:这里在 做ON 判断时 已经对name 字段进行匹配了,这就好比我在进行一组表更新操作的时候的锁表状态,所以想更新NAME 便不能用NAME 做条件判断。

思考:错误二引发对错误一的思考

假使我在做ON判断的时候用的是表的主键,然后我想做UPDATE 操作的时候如果是on 里面的条件字段,也就是说 要更新的是  数据库 中 表的主键 ,这也就违背了  数据库的主键约束条件。因此,从错误二去反推错误一,就自然好理解了。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏落花落雨不落叶

canvas画简单电路图

60911
来自专栏张善友的专栏

Miguel de Icaza 细说 Mix 07大会上的Silverlight和DLR

Mono之父Miguel de Icaza 详细报道微软Mix 07大会上的Silverlight和DLR ,上面还谈到了Mono and Silverligh...

2707
来自专栏魂祭心

原 canvas绘制clock

4054
来自专栏我和未来有约会

Silverlight第三方控件专题

这里我收集整理了目前网上silverlight第三方控件的专题,若果有所遗漏请告知我一下。 名称 简介 截图 telerik 商 RadC...

3995
来自专栏Ceph对象存储方案

Luminous版本PG 分布调优

Luminous版本开始新增的balancer模块在PG分布优化方面效果非常明显,操作也非常简便,强烈推荐各位在集群上线之前进行这一操作,能够极大的提升整个集群...

3105
来自专栏hbbliyong

WPF Trigger for IsSelected in a DataTemplate for ListBox items

<DataTemplate DataType="{x:Type vm:HeaderSlugViewModel}"> <vw:HeaderSlug...

4064
来自专栏跟着阿笨一起玩NET

c#实现打印功能

2702
来自专栏闻道于事

js登录滑动验证,不滑动无法登陆

js的判断这里是根据滑块的位置进行判断,应该是用一个flag判断 <%@ page language="java" contentType="text/html...

6768
来自专栏菩提树下的杨过

Flash/Flex学习笔记(23):运动学原理

先写一个公用的小球类Ball: package{ import flash.display.Sprite; //小球 类 public class B...

25210
来自专栏大内老A

The .NET of Tomorrow

Ed Charbeneau(http://developer.telerik.com/featured/the-net-of-tomorrow/) Exciti...

31410

扫码关注云+社区