What is wrong with my update statement with a join in Oracle?

后端 未结 8 1484
梦谈多话
梦谈多话 2021-01-02 20:29

I am working with an Oracle 10g Database.

I have the following two tables:

T_DEBTOR :
    - ID_DEBTOR
    - HEADER
T_ELEMENT :
    -         


        
8条回答
  •  长情又很酷
    2021-01-02 20:35

    Good question.

    To simulate your situation, I've created sample tables:

    SQL> create table t_debtor(id_debtor,header)
      2  as
      3  select 1, 'Header 1' from dual union all
      4  select 2, null from dual union all
      5  select 3, 'Header 3' from dual
      6  /
    
    Tabel is aangemaakt.
    
    SQL> create table t_element (id_element,id_debtor,insurer)
      2  as
      3  select 1, 1, 'to be updated' from dual union all
      4  select 2, 1, 'to be updated' from dual union all
      5  select 3, 2, 'not to be updated' from dual union all
      6  select 4, 2, 'not to be updated' from dual union all
      7  select 5, 3, 'to be updated' from dual
      8  /
    
    Tabel is aangemaakt.
    

    And with your current update statement, the problem becomes clear: the "not to be updated" values are set to NULL:

    SQL> update
      2      T_ELEMENT elt
      3      set elt.INSURER = (
      4          select HEADER
      5              from T_DEBTOR debtor
      6              where
      7                  debtor.HEADER is not null
      8                  and debtor.ID_DEBTOR = elt.ID_DEBTOR)
      9  /
    
    5 rijen zijn bijgewerkt.
    
    SQL> select * from t_element
      2  /
    
    ID_ELEMENT  ID_DEBTOR INSURER
    ---------- ---------- -----------------
             1          1 Header 1
             2          1 Header 1
             3          2
             4          2
             5          3 Header 3
    
    5 rijen zijn geselecteerd.
    

    The best way to do this update, is to update a join of both tables. There are some restrictions however:

    SQL> rollback
      2  /
    
    Rollback is voltooid.
    
    SQL> update ( select elt.insurer
      2                , dtr.header
      3             from t_element elt
      4                , t_debtor dtr
      5            where elt.id_debtor = dtr.id_debtor
      6              and dtr.header is not null
      7         )
      8     set insurer = header
      9  /
       set insurer = header
           *
    FOUT in regel 8:
    .ORA-01779: cannot modify a column which maps to a non key-preserved table
    

    With the bypass ujvc hint, we can circumvent this restriction. But it is not advisable to do so unless you know really really sure that t_debtor.id_debtor is unique.

    SQL> update /*+ bypass_ujvc */
      2         ( select elt.insurer
      3                , dtr.header
      4             from t_element elt
      5                , t_debtor dtr
      6            where elt.id_debtor = dtr.id_debtor
      7              and dtr.header is not null
      8         )
      9     set insurer = header
     10  /
    
    3 rijen zijn bijgewerkt.
    
    SQL> select * from t_element
      2  /
    
    ID_ELEMENT  ID_DEBTOR INSURER
    ---------- ---------- -----------------
             1          1 Header 1
             2          1 Header 1
             3          2 not to be updated
             4          2 not to be updated
             5          3 Header 3
    
    5 rijen zijn geselecteerd.
    

    It's better to just add a primary key. You'll probably have this one already in place:

    SQL> rollback
      2  /
    
    Rollback is voltooid.
    
    SQL> alter table t_debtor add primary key (id_debtor)
      2  /
    
    Tabel is gewijzigd.
    
    SQL> update ( select elt.insurer
      2                , dtr.header
      3             from t_element elt
      4                , t_debtor dtr
      5            where elt.id_debtor = dtr.id_debtor
      6              and dtr.header is not null
      7         )
      8     set insurer = header
      9  /
    
    3 rijen zijn bijgewerkt.
    
    SQL> select * from t_element
      2  /
    
    ID_ELEMENT  ID_DEBTOR INSURER
    ---------- ---------- -----------------
             1          1 Header 1
             2          1 Header 1
             3          2 not to be updated
             4          2 not to be updated
             5          3 Header 3
    
    5 rijen zijn geselecteerd.
    

    Regards, Rob.

提交回复
热议问题