oracle 多表連合修改----BYPASS_UJVC(轉)

warehouse發表於2010-12-21

原文連線:

http://blog.chinaunix.net/u/28618/showart_347103.html

這就是oracle中支援的多表關聯更新中的語法,這種語法比使用子查詢更新效率更高。

[@more@]
方法一:
ORA-01779: cannot modify a column which maps to a non-key-preserved table
例如,使用以下的更新查詢就會出現該錯誤。

CREATE TABLE test1 ( id integer primary key, num integer );
INSERT INTO test1 VALUES (1,0);
INSERT INTO test1 VALUES (2,0);
INSERT INTO test1 VALUES (3,0);
INSERT INTO test1 VALUES (4,0);
CREATE TABLE test2 ( id integer, num integer, upd integer );
INSERT INTO test2 VALUES (1,10, 0);
INSERT INTO test2 VALUES (2,20, 1);
UPDATE ( SELECT t1.id id1, t1.num num1, t2.id id2, t2.num num2
FROM test1 t1, test2 t2 WHERE t1.id=t2.id AND t2.upd=1 )
SET num1=num2; ORA-01779: cannot modify a column which maps to a non-key-preserved table
這個錯誤的意思是,子查詢的結果中,更新資料來源(test2)的內容不唯一,導致被更新物件(test1)中的一行可能對應資料來源(test2)中的多行。 本例中,test2表的id不唯一,因此test2表中可能存在id相同但是num不相同的資料,這種資料是無法用來更新 test1 的。

解決方法就是保證資料來源的唯一性,例如本例中可以為test2.id建立一個唯一索引:

CREATE UNIQUE INDEX test2_idx_001 ON test2 (id);
之後上面的更新就可以執行了。

另外也可以強制 Oracle 執行,方法是加上 BYPASS_UJVC 註釋。

UPDATE
( SELECT /*+ BYPASS_UJVC */ t1.id id1, t1.num num1, t2.id id2, t2.num num2
FROM test1 t1, test2 t2
WHERE t1.id=t2.id AND t2.upd=1 )
SET num1=num2;
BYPASS_UJVC的作用是跳過Oracle的鍵檢查。 這樣雖然能夠執行了,但是如果test2中存在不唯一的資料,test1就會被更新多次而導致意想不到的結果。

方法二:
update (select /*+ BYPASS_UJVC */ name , rname
from table1 t1, table2 t2
where t1.id = t2.id
and t1.id is not null)
set name = rname

-- 對應的MERGE格式()
-- 只更新不插入的語法 只有10g才能支援,9i還不能支援!
-- 因此如果是9i,則還是要使用對檢視的UPDATE語句
MERGE INTO david_1 T1
USING david_2 T2
ON (T1.A = T2.A AND t1.a <=2)
WHEN MATCHED THEN
UPDATE
SET T1.B = T2.B
--WHEN NOT MATCHED THEN NOTHING

來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/19602/viewspace-1043411/,如需轉載,請註明出處,否則將追究法律責任。

相關文章