At my current assignment I needed to create an update statement to copy data from one table to another. Quite a simple task, I would say, but an error or actually a constraint violation sent me the wrong way in finding my solution.
Suppose I have two tables:
CREATE TABLE a ( mykey NUMBER , thisvalue VARCHAR2(20) NOT NULL ) /
CREATE TABLE b ( mykey NUMBER , thatvalue VARCHAR2(20) NOT NULL ) /
and some data in them:
INSERT INTO a (mykey,thisvalue) VALUES (1,'Larry'); INSERT INTO a (mykey,thisvalue) VALUES (2,'Bryn'); INSERT INTO a (mykey,thisvalue) VALUES (3,'Steven'); INSERT INTO a (mykey,thisvalue) VALUES (4,'Patrick'); INSERT INTO b (mykey,thatvalue) VALUES (1,'Larry Ellison'); INSERT INTO b (mykey,thatvalue) VALUES (2,'Bryn Llewellyn'); INSERT INTO b (mykey,thatvalue) VALUES (3,'Steven Feuerstein'); COMMIT /
Now I want to update the values in table a with the values of table b. My first idea was to write a statement like this:
UPDATE a SET a.thisvalue = (select b.thatvalue FROM b WHERE b.mykey = a.mykey) /
but this statement led to the following error:
ORA-01407: cannot update ("DEMO"."A"."THISVALUE") to NULL
No problem, I thought, if the new value is somehow NULL, then just use the old value:
UPDATE a SET a.thisvalue = (select NVL(b.thatvalue, a.thisvalue) FROM b WHERE b.mykey = a.mykey) /
but this still resulted in the same error:
ORA-01407: cannot update ("DEMO"."A"."THISVALUE") to NULL
Then it dawned upon me. For mykey=4 there would be no match in table B, which resulted in no row returned, hence a NULL value.
The error given is absolutely correct, but it sent me the wrong way in finding a solution. If I would have gotten a NO DATA FOUND error, I would have known right away what was the problem.
The solution was pretty easy, just update the rows that have a matching row in table B:
UPDATE a SET a.thisvalue = (select b.thatvalue FROM b WHERE b.mykey = a.mykey) WHERE EXISTS (select b.thatvalue FROM b WHERE b.mykey = a.mykey) /
Another solution might be using the MERGE statement:
MERGE INTO a USING (select b.mykey, b.thatvalue FROM b) b ON (a.mykey = b.mykey) WHEN MATCHED THEN UPDATE SET a.thisvalue = b.thatvalue /
If the subselect results in more than one row you get an equivalent of the TOO_MANY_ROWS exception, but if the subselect results in no rows you don’t get the NO_DATA_FOUND (or equivalent) exception.
Or
UPDATE a
SET a.thisvalue = NVL( (select b.thatvalue
FROM b
WHERE b.mykey = a.mykey)
, a.thisvalue)
/
In the case of more than one row, whether the MERGE returns an exception can depend on whether the order and value of the data being results in more than one changed value to any single row.
e.g. A -> A -> B = no error
but A -> B -> C (or A) = error