日期:2014-05-16  浏览次数:20547 次

使用merge语句更新数据库中的记录
    在平时更新数据时,经常有这样一种更新,即将表中的数据与源表对比,如果存在记录,则根据源表中的值更新目标表中的数据,如果不存在的话,则新增入目标表中。我们当然可以使用两条语句来处理这类数据。但这其中有可能会出现异常。因此,Oracle在9i版本新增了MERGE语句,来合并UPDATE和INSERT语句。

    通过MERGE语句,根据一张表或子查询的连接条件对另外一张表进行查询,连接条件匹配上的进行UPDATE,无法匹配的执行INSERT。这个语法仅需要一次全表扫描就完成了全部工作,执行效率要高于INSERT+UPDATE。

    具体语法为:
    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的基本用法 (此处目标表和源表表结构相同)
  1) matched 和not matched 同时使用
   merge into 目标表  a
     using 源表 b on (关联条件 a.字段1 = b.字段1)
   when MATCHED then
        update set a.字段2=b.字段2,......
   when NOT MATCHED then
        insert(a.字段2,a.字段3)
        values(b.字段2,b.字段3);
  2) 只有not matched clause,也就是只插入不更新
   merge into 目标表 a
     using  源表b on (关联条件 a.字段1 = b.字段1)  
   when NOT MATCHED then
       insert(a.字段2,a.字段3)
        values(b.字段2,b.字段3);

  3) 只有matched clause, 也就是只更新不插入
   merge into 目标表  a
     using 源表 b on (关联条件 a.字段1 = b.字段1)
   when MATCHED then
        update set a.字段2=b.字段2,......
   
另外,merge命令的update部分可以包含一个delete子句。delete子句(有其自己的where子句)可以删除目标表中被merge更新的行。delete...where子句可以计算更新值,而不是目标表中的初始值。如果目标表中的行符合delete...where条件但不在merge所作用(就如on条件所定义的)的行集范围内,就不会删除。
    具体语法为:
       using 源表 b on (关联条件 a.字段1 = b.字段1)
   when MATCHED then
        update set a.字段2=b.字段2,......
        delete where (a.字段5=b.字段5) ;

   注意:此处的源表可以不仅仅是一张表,也可以是查询出来的结果集。此命令在db2中也同样适用。