问题 Oracle sql merge要插入和删除但不能更新


有没有办法使用oracle merge来插入和删除但不更新?

我有一个表,表示与另一个表中的单个行相关的一组值。我可以通过删除所有值并添加新的集合来更改值集,或者通过有选择地删除一些值并添加其他值来更改值,但我有兴趣在可能的情况下将其作为单个语句。

这是一个更新的工作示例。为了完成这项工作,我不得不补充一下 dummy 以便可以更新的列不在 on 条件。有没有办法只删除和插入没有虚拟列来更新?

没有专栏 on 条件可能在 update set 列表,即使它实际上没有更新。

create table every_value ( the_value varchar2(32) );
create table paired_value ( the_id number, a_value varchar2(32) , dummy number default 0 );
-- the_id is a foreign_key to a row in another table

insert into every_value ( the_value ) values ( 'aaa' );
insert into every_value ( the_value ) values ( 'abc' );
insert into every_value ( the_value ) values ( 'ace' );
insert into every_value ( the_value ) values ( 'adg' );
insert into every_value ( the_value ) values ( 'aei' );
insert into every_value ( the_value ) values ( 'afk' );

-- pair ace and afk with id 3
merge into paired_value p using every_value e
on ( p.the_id = 3 and p.a_value = e.the_value )
when matched then update set dummy=dummy+1
delete where a_value not in ('ace','afk')
when not matched then insert (the_id,a_value)
values (3,e.the_value)
where e.the_value in ('ace','afk');

-- pair ace and aei with id 3
-- should remove afk, add aei, do nothing with ace
merge into paired_value p using every_value e
on ( p.the_id = 3 and p.a_value = e.the_value )
when matched then update set dummy = dummy+1
delete where a_value not in ('ace','aei')
when not matched then insert (the_id,a_value)
values (3,e.the_value)
where e.the_value in ('ace','aei');

-- pair aaa and adg with id 4
merge into paired_value p using every_value e
on ( p.the_id = 4 and p.a_value = e.the_value )
when matched then update set dummy = dummy+1
delete where a_value not in ('aaa','adg')
when not matched then insert (the_id,a_value)
values (4,e.the_value)
where e.the_value in ('aaa','adg');

select * from paired_value;

我已经在oracle 10g中尝试了这个,并且有了这个 sqlfiddle,oracle 11g。


12769
2017-07-17 20:28


起源



答案:


不,您无法删除merge命令尚未更新的行。
这是文档: http://docs.oracle.com/cd/B28359_01/server.111/b28286/statements_9016.htm

指定DELETE where_clause以清除表中的数据   填充或更新它。 受此条款影响的唯一行是   目标表中由合并更新的那些行   手术。 DELETE WHERE条件不会评估更新的值   由UPDATE SET ... WHERE评估的原始值   条件。如果目标表的一行符合DELETE   条件但不包含在ON子句定义的连接中,   然后它不会被删除。在目标上定义的任何删除触发器   将为每行删除激活表。

那意味着,那行 必须 得到更新。 Hovewer,在UPDATE使用与DELETE之后使用的相同WHERE子句之后,您不需要更新所有行

when matched then update set dummy=dummy
    where a_value not in ('ace','afk')
delete 
    where a_value not in ('ace','afk')

15
2017-07-17 20:57



我很确定没有比使用虚拟列更好的方法,但我不会为此添加虚拟列。好吧。 - drawnonward


答案:


不,您无法删除merge命令尚未更新的行。
这是文档: http://docs.oracle.com/cd/B28359_01/server.111/b28286/statements_9016.htm

指定DELETE where_clause以清除表中的数据   填充或更新它。 受此条款影响的唯一行是   目标表中由合并更新的那些行   手术。 DELETE WHERE条件不会评估更新的值   由UPDATE SET ... WHERE评估的原始值   条件。如果目标表的一行符合DELETE   条件但不包含在ON子句定义的连接中,   然后它不会被删除。在目标上定义的任何删除触发器   将为每行删除激活表。

那意味着,那行 必须 得到更新。 Hovewer,在UPDATE使用与DELETE之后使用的相同WHERE子句之后,您不需要更新所有行

when matched then update set dummy=dummy
    where a_value not in ('ace','afk')
delete 
    where a_value not in ('ace','afk')

15
2017-07-17 20:57



我很确定没有比使用虚拟列更好的方法,但我不会为此添加虚拟列。好吧。 - drawnonward


我发现你可以将列设置为自己:

MERGE ...
WHEN MATCHED THEN 
   UPDATE SET a_value = a_value WHERE a_value not in ('ace','afk')
   DELETE WHERE a_value not in ('ace','afk')

这消除了对虚拟列的需要。


1
2018-02-20 23:46



你的ON条件是as_value吗?如果列处于ON状态,这对我来说在11g中不起作用我得到“ON子句中引用的列无法更新”错误,即使更新正在将值设置为自身。 - Marquez
那是对的。根据oracle文档,您无法更新属于ON子句的列。如果你考虑一下,这是有道理的。但是您可以在更新中使用不同的非虚拟列;没有尝试过它我不知道删除子句是否仍然会触发相同的错误?例如。 SET b_value = b_value其中a_value不在('ace','afk')DELETE,其中a_value不在('ace','afk')。 - datico