Oracle update 多表

本文介绍了五种不同的SQL更新方法,包括直接更新、使用EXISTS、利用游标、MERGE语句等,解决了更新过程中遇到的问题,并提供了详细的实现步骤。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

$ sqlplus user/pass

SQL*Plus: Release 9.2.0.6.0 - Production on Wed Aug 2 17:38:39 2006

Copyright (c) 1982, 2002, Oracle Corporation.  All rights reserved.


Connected to:
Oracle9i Enterprise Edition Release 9.2.0.6.0 - 64bit Production
With the Partitioning, OLAP and Oracle Data Mining options
JServer Release 9.2.0.6.0 - Production

SQL> select * from wwm2;        --要更新的表

TOWN                         ID
-------------------- ----------
222                         222
111                         111
ww'jj                       111
llll                       1111
dddd                       2222
lllldf                      111
lllldf                      111
dsafdf                      111
3435                        111
ljjjjj                      222
dsafdf                      111
TOWN                         ID
-------------------- ----------
3435                        111
ljjjjj                      222

SQL> select * from wwm5;            --更新的条件表
TOWN                         ID
-------------------- ----------
lllldf                      111
test                       9984
SQL> select wwm2.* from wwm2,wwm5 where wwm2.id=wwm5.id
  2  /
TOWN                         ID
-------------------- ----------
111                         111
ww'jj                       111
lllldf                      111
lllldf                      111
dsafdf                      111
3435                        111
dsafdf                      111
3435                        111
8 rows selected.
所以,每次需要更新8条数据就是正确的.
相信程序员是通过以下类似的SQL更新的,这是错误的,因为没有加WHERE
SQL>  update wwm2 set wwm2.town=(select wwm5.town from wwm5 where wwm5.id=wwm2.id)
  2  /
13 rows updated.
SQL> select * from wwm2;
TOWN                         ID
-------------------- ----------
                            222
lllldf                      111
lllldf                      111
                           1111
                           2222
lllldf                      111
lllldf                      111
lllldf                      111
lllldf                      111
                            222
lllldf                      111
TOWN                         ID
-------------------- ----------
lllldf                      111
                            222
13 rows selected.
可以看到13条记录被更新,符合条件的更新正确,不符合条件的也更新为NULL.以下是正确的方法
方法一:
SQL> update wwm2
  2  set town=(select town from wwm5 where wwm5.id=wwm2.id)
  3  where id=(select wwm5.id from wwm5 where wwm5.id=wwm2.id)
  4  /
8 rows updated.
方法二:    与方法一道理相同,这里需要掌握EXIST的相关用法.
SQL> update wwm2
   set town=(select town from wwm5 where wwm5.id=wwm2.id)
   where exists (select 1 from wwm5 where wwm5.id=wwm2.id)
8 rows updated.
方法三:
SQL> update (select a.town atown,a.id aid,b.town btown,b.id bid from wwm2 a,wwm5 b where a.id=b.id)
  2  set atown=btown
  3  /
set atown=btown
    *
ERROR at line 2:
ORA-01779: cannot modify a column which maps to a non key-preserved table
  1* alter table wwm5 add primary key (id)
SQL> /
Table altered.
  1  update (select a.town atown,a.id aid,b.town btown,b.id bid from wwm2 a,wwm5 b where a.id=b.id)
  2*  set atown=btown
SQL> /
8 rows updated.
这种方法的局限性就是需要PRIMARY 的支持.
方法四:
  1  declare
  2  cursor cur_wwm is select town,id from wwm5;
  3  begin
  4     for my_wwm in cur_wwm loop
  5     update wwm2 set town=my_wwm.town
  6     where id=my_wwm.id;
  7     end loop;
  8* end;
SQL> /
PL/SQL procedure successfully completed.
SQL> select * from wwm2;
TOWN                         ID
-------------------- ----------
222                         222
lllldf                      111
lllldf                      111
llll                       1111
dddd                       2222
lllldf                      111
lllldf                      111
lllldf                      111
lllldf                      111
ljjjjj                      222
lllldf                      111
TOWN                         ID
-------------------- ----------
lllldf                      111
ljjjjj                      222
这个方法是最灵活的了.
方法五:
注意,方法五只能适用于WWM5是WWM2的子集的时候.
  1   merge into wwm2
  2   using (select town,id from wwm5) b
  3   on (wwm2.id=b.id)
  4   when matched then update set town=b.town
  5* when not matched then insert (town,id) values (null,null)
SQL> /
9 rows merged.
SQL> select * from wwm2;
TOWN                         ID
-------------------- ----------
                                  ---注意这个地方,被插入了一个空值.因为WWM5的ID=9984在WWM2中不能匹配,根本原因是ORACLE9必须有WHEN NOT MATCHED子句,但是ORACLE10可以不许要,也就是ORACLE10可以不写WHEN NOT MATCHED ,就不必插入NULL值了,为解决这个问题,下一步会DELETE WWM5的ID=9984,这样一来就不会执行WHEN NOT MATCHED
222                         222
lllldf                      111
lllldf                      111
llll                       1111
dddd                       2222
lllldf                      111
lllldf                      111
lllldf                      111
lllldf                      111
ljjjjj                      222
TOWN                         ID
-------------------- ----------
lllldf                      111
lllldf                      111
ljjjjj                      222
14 rows selected.
SQL> delete from wwm5 where id=9984;
1 row deleted.
SQL>  1   merge into wwm2                            
SQL>   2   using (select town,id from wwm5) b
SQL>   3   on (wwm2.id=b.id)
SQL>   4   when matched then update set town=b.town
SQL>   5* when not matched then insert (town,id) values (null,null)
SQL> /
8 rows merged.
 
       以上就是5种关连更新的例子了,希望能给开发人员解惑
### 更新操作 在Oracle数据库中,直接通过单一SQL语句实现的同时更新并非原生支持的功能。然而,可以通过种方式间接达成这一目的。 #### 使用子查询的方式进行更新 当需要基于另一张中的数据来更新当前时,可以采用带有子查询的UPDATE语句。例如: ```sql UPDATE employees e SET salary = ( SELECT AVG(salary) FROM departments d WHERE d.department_id = e.department_id ); ``` 此命令会将员工工资设置为其所在部门平均薪资[^1]。 #### 利用MERGE语句完成跨更新 对于更复杂的场景,比如依据关联条件同步修改两个或更格的内容,则推荐使用`MERGE INTO`结构。这种方式允许在一个事务内处理个目标之间的匹配记录并实施相应的动作(插入/更新)。下面是一个简单的例子说明如何利用merge来进行有条件的选择性更新: ```sql MERGE INTO target_table t USING source_table s ON (t.key_column = s.key_column) WHEN MATCHED THEN UPDATE SET t.column_to_update = s.value_from_source; ``` 这里的关键在于定义好ON子句里的连接逻辑以及具体要改变哪些字段值[^3]。 #### PL/SQL块内的批量更新 另外一种常见做法就是在匿名PL/SQL程序单元里编写循环控制流代码逐条访问待更改对象集合,并调用标准DML指令逐一作用于各成员之上;这种方法特别适合那些涉及复杂业务规则判断的情况。示例代码如下所示: ```plsql BEGIN FOR rec IN (SELECT * FROM some_view_or_query) LOOP IF condition_based_on_rec THEN UPDATE another_table at SET ... WHERE id = rec.id; INSERT INTO log_table lt VALUES (...); -- 可选的日志记录 DELETE FROM third_table tt WHERE ref_id = rec.ref_id; -- 或者其它DML操作 END IF; END LOOP; COMMIT; EXCEPTION WHEN OTHERS THEN ROLLBACK; END; / ``` 上述脚本展示了怎样构建一个完整的事务流程,在其中包含了不同类型的变更活动[^4]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值