Oracle中試圖對一個子查詢進行更新時可能會出現ORA-01779錯誤。該錯誤的內容為:
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就會被更新多次而導致意想不到的結果。