UPSERT操作在表中更新或插入一行,这取决于表中是否已经有与数据匹配的行:
if table t has a row exists that has key X:
update t set mystuff... where mykey=X
else
insert into t mystuff...
既然Oracle没有特定的UPSERT语句,那么最好的方法是什么呢?
UPSERT操作在表中更新或插入一行,这取决于表中是否已经有与数据匹配的行:
if table t has a row exists that has key X:
update t set mystuff... where mykey=X
else
insert into t mystuff...
既然Oracle没有特定的UPSERT语句,那么最好的方法是什么呢?
当前回答
从http://www.praetoriate.com/oracle_tips_upserts.htm:
在Oracle9i中,UPSERT可以在一条语句中完成这个任务:
INSERT
FIRST WHEN
credit_limit >=100000
THEN INTO
rich_customers
VALUES(cust_id,cust_credit_limit)
INTO customers
ELSE
INTO customers SELECT * FROM new_customers;
其他回答
从http://www.praetoriate.com/oracle_tips_upserts.htm:
在Oracle9i中,UPSERT可以在一条语句中完成这个任务:
INSERT
FIRST WHEN
credit_limit >=100000
THEN INTO
rich_customers
VALUES(cust_id,cust_credit_limit)
INTO customers
ELSE
INTO customers SELECT * FROM new_customers;
关于两种解决方案的说明如下:
1)插入,如果异常则更新,
or
2)更新,如果sql%rowcount = 0,则插入
先插入还是先更新的问题也取决于应用程序。您是否期待更多的插入或更新?最有可能成功的应该先走。
如果你选错了,你会得到一堆不必要的索引读取。不是什么大问题,但仍然值得考虑。
多年来,我一直在使用第一个代码示例。注意notfound而不是count。
UPDATE tablename SET val1 = in_val1, val2 = in_val2
WHERE val3 = in_val3;
IF ( sql%notfound ) THEN
INSERT INTO tablename
VALUES (in_val1, in_val2, in_val3);
END IF;
下面的代码可能是新的和改进的代码
MERGE INTO tablename USING dual ON ( val3 = in_val3 )
WHEN MATCHED THEN UPDATE SET val1 = in_val1, val2 = in_val2
WHEN NOT MATCHED THEN INSERT
VALUES (in_val1, in_val2, in_val3)
在第一个示例中,更新执行索引查找。为了更新右边行,它必须这样做。Oracle打开一个隐式游标,我们使用它来包装相应的插入,这样我们就知道只有当键不存在时才会发生插入。但是插入是一个独立的命令,它必须进行第二次查找。我不知道merge命令的内部工作原理,但由于该命令是一个单独的单元,Oracle可以通过单个索引查找执行正确的插入或更新。
我认为合并更好,当你有一些处理要做,这意味着从一些表中获取数据,并更新一个表,可能插入或删除行。但是对于单行情况,您可以考虑第一个情况,因为语法更常见。
我想要Grommit的答案,除了它需要欺骗的值。我在可能出现一次的地方找到了解决方案:http://forums.devshed.com/showpost.php?p=1182653&postcount=2
MERGE INTO KBS.NUFUS_MUHTARLIK B
USING (
SELECT '028-01' CILT, '25' SAYFA, '6' KUTUK, '46603404838' MERNIS_NO
FROM DUAL
) E
ON (B.MERNIS_NO = E.MERNIS_NO)
WHEN MATCHED THEN
UPDATE SET B.CILT = E.CILT, B.SAYFA = E.SAYFA, B.KUTUK = E.KUTUK
WHEN NOT MATCHED THEN
INSERT ( CILT, SAYFA, KUTUK, MERNIS_NO)
VALUES (E.CILT, E.SAYFA, E.KUTUK, E.MERNIS_NO);
试试这个,
insert into b_building_property (
select
'AREA_IN_COMMON_USE_DOUBLE','Area in Common Use','DOUBLE', null, 9000, 9
from dual
)
minus
(
select * from b_building_property where id = 9
)
;