把数据从一个表复制到另一个表,插入新数据或替换掉老数据是每一个ORACLE DBA都会经常碰到的问题。在ORACLE9i以前的年代,我们要先查找是否存在老数据,如果有用UPDATE替换,否则用INSERT语句插入,其间少不了还有一些标记变量等等,繁琐的很。现在ORACLE9i专为这种情况提供了MERGE语句,使这一工作变得异常轻松,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;
例如:
MERGE INTO tdest d
USING tsrc s
ON (s.srckey = d.destkey)
WHEN MATCHED THEN
UPDATE SET d.destdata = d.destdata + s.srcdata
WHEN NOT MATCHED THEN
INSERT (destkey, destdata) VALUES (srckey, srcdata)
一条语句代替了原来的一段复杂语句,但是使用MERGE也是有要注意的地方的,看下面的例子: CREATE TABLE tdest(destkey KEYTYE,destdata DATATYPE);
CREATE TABLE tsrc(srckey KEYTYE,srcdata DATATYPE); --insert some rows into tsrc here.
MERGE INTO tdest d
USING tsrc s
ON (d.destkey = s.srckey)
WHEN MATCHED THEN
UPDATE SET d.destkey = s.srckey
WHEN NOT MATCHED THEN
INSERT (d.destkey, d.destdata) VALUES (s.srckey, s.srcdata);
运行以上代码,你可能会惊讶得得到如下结果:
ORA-00904: invalid column name Cause: The column name entered is either missing or invalid. Action: Enter a valid column name.
你一开始会觉得很蹊跷,于是到处找自己哪里写错了,其实你没有写错,呵呵,是ORACLE错了。原来,其中关键是那个UPDATE语句,不能更新在ON语句中引用得那些列,这可以说是一个限制,也可以说是一个BUG,因为毕竟ORACLE的正式DOCUMENT里并没有提有这一限制。metalink里你可以找[BUG:2124282] 对这一问题的描述,可见ORACLE确实把它当成了一个BUG对待,或许,后续的版本就没有这一限制了,但是目前,如果你想从ORACLE工程师提供MERGE功能中获益的话,还是要记住这个小BUG的。