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就會被更新多次而導致意想不到的結果。