Sql Oracle Merge delete子句,其中数据在目标中,但不在源中

Sql Oracle Merge delete子句,其中数据在目标中,但不在源中,sql,oracle,plsql,oracle12c,Sql,Oracle,Plsql,Oracle12c,试图从目标表中删除不在源表中的记录 对象类型 收集类型 存储过程 目前,我可以从一个被截断的表开始,插入部分可以工作。当再次运行时,更新部分工作。 但是,当从tmpPList中删除记录时,即不再在源数据中,该记录将保留在目标表中。在目标表中,仅删除与ON子句和DELETE WHERE匹配的行 您可以使用查询与TMP完全联接表,并使用标志删除不匹配的行 create or replace PROCEDURE testProc(tmpPList IN P_REC_LIST) AS BEGIN I

试图从目标表中删除不在源表中的记录

对象类型

收集类型

存储过程

目前,我可以从一个被截断的表开始,插入部分可以工作。当再次运行时,更新部分工作。 但是,当从tmpPList中删除记录时,即不再在源数据中,该记录将保留在目标表中。

在目标表中,仅删除与ON子句和DELETE WHERE匹配的行

您可以使用
查询
TMP
完全联接
表,并使用标志删除不匹配的行

create or replace  PROCEDURE testProc(tmpPList IN P_REC_LIST)
AS
BEGIN
 IF tmpPList.count > 0 THEN
MERGE INTO targetTable tt
        USING ( with abc as
                 (
                  select * from TABLE(tmpPList)
                  )
                  select COALESCE(abc.attr1,t.attr1) as attr1,
                                     abc.attr2,CASE WHEN abc.attr1 IS NULL
                                           THEN 'Y' ELSE 'N' END 
                                          match_flag FROM abc
                          FULL JOIN targetTable t ON t.attr1 = abc.attr1
               ) tmp
        ON (tt.Attr1 = tmp.attr1)
      WHEN MATCHED THEN
         UPDATE SET tt.Attr2 = tmp.Attr2 
        DELETE WHERE match_flag = 'Y' --This flag to identify the unmatched row
      WHEN NOT MATCHED THEN
        INSERT (Attr1)
        VALUES (tmp.Attr1);
  END IF;
END testProc;
/

在a中,仅删除目标表中与ON子句和DELETE WHERE匹配的行

您可以使用
查询
TMP
完全联接
表,并使用标志删除不匹配的行

create or replace  PROCEDURE testProc(tmpPList IN P_REC_LIST)
AS
BEGIN
 IF tmpPList.count > 0 THEN
MERGE INTO targetTable tt
        USING ( with abc as
                 (
                  select * from TABLE(tmpPList)
                  )
                  select COALESCE(abc.attr1,t.attr1) as attr1,
                                     abc.attr2,CASE WHEN abc.attr1 IS NULL
                                           THEN 'Y' ELSE 'N' END 
                                          match_flag FROM abc
                          FULL JOIN targetTable t ON t.attr1 = abc.attr1
               ) tmp
        ON (tt.Attr1 = tmp.attr1)
      WHEN MATCHED THEN
         UPDATE SET tt.Attr2 = tmp.Attr2 
        DELETE WHERE match_flag = 'Y' --This flag to identify the unmatched row
      WHEN NOT MATCHED THEN
        INSERT (Attr1)
        VALUES (tmp.Attr1);
  END IF;
END testProc;
/

完美!正是我需要的。非常感谢。当我在USING子句中的物化视图上添加左连接并尝试在updatewhere子句中检查多个列值时,DELETE子句似乎失败了,并且没有删除记录。您是否能够提供其他帮助?@XmlevolentX:这回答了您最初的问题。如果你有更多的问题,你应该考虑用最新的细节询问新的,而不是不接受一个被接受的答案。谢谢你的建议。@KaushikNayak我试图在我的例子中复制这个解决方案,但我不明白“匹配标志”是从哪里来的?我在SQL中遇到以下错误:“match_flag error report-Unknown Command”完美!正是我需要的。非常感谢。当我在USING子句中的物化视图上添加左连接并尝试在updatewhere子句中检查多个列值时,DELETE子句似乎失败了,并且没有删除记录。您是否能够提供其他帮助?@XmlevolentX:这回答了您最初的问题。如果你有更多的问题,你应该考虑用最新的细节询问新的,而不是不接受一个被接受的答案。谢谢你的建议。@KaushikNayak我试图在我的例子中复制这个解决方案,但我不明白“匹配标志”是从哪里来的?我在SQL中遇到以下错误:“匹配标志错误报告-未知命令”
PROCEDURE testProc(tmpPList IN P_REC_LIST, resultCursor out sys_refcursor)

IF tmpPList.count > 0 THEN
MERGE INTO [targetTable] TT
        USING (SELECT abc.Attr1 abc.Attr2 FROM TABLE(tmpPList) abc) TMP
        ON (TT.Attr1 = TMP.Attr1)
      WHEN MATCHED THEN
        UPDATE SET TT.Attr2 = TMP.Attr2 
        DELETE WHERE TT.Attr1 NOT IN (SELECT Attr1 FROM TABLE(tmpPList))
      WHEN NOT MATCHED THEN
        INSERT (Attr1)
        VALUES (TMP.Attr1);
END IF;
create or replace  PROCEDURE testProc(tmpPList IN P_REC_LIST)
AS
BEGIN
 IF tmpPList.count > 0 THEN
MERGE INTO targetTable tt
        USING ( with abc as
                 (
                  select * from TABLE(tmpPList)
                  )
                  select COALESCE(abc.attr1,t.attr1) as attr1,
                                     abc.attr2,CASE WHEN abc.attr1 IS NULL
                                           THEN 'Y' ELSE 'N' END 
                                          match_flag FROM abc
                          FULL JOIN targetTable t ON t.attr1 = abc.attr1
               ) tmp
        ON (tt.Attr1 = tmp.attr1)
      WHEN MATCHED THEN
         UPDATE SET tt.Attr2 = tmp.Attr2 
        DELETE WHERE match_flag = 'Y' --This flag to identify the unmatched row
      WHEN NOT MATCHED THEN
        INSERT (Attr1)
        VALUES (tmp.Attr1);
  END IF;
END testProc;
/