oracle关联更新sql
恶作剧 databaseoracle
# 关联更新三种方式
准备好 表1,表2
select * from t1;
1
| FNAME | FMONEY |
|---|---|
| A | 20 |
| B | 30 |
select * from t2;
1
| FNAME | FMONEY |
|---|---|
| A | 100 |
| C | 40 |
| D | 50 |
需求:参考 T2,修改 T1 表,修改条件为两表的 FNAME 列内容一致。
# 方式一 UPDATE
常见陷阱:
UPDATE T1
SET T1.FMONEY = (select T2.FMONEY from t2 where T2.FNAME = T1.FNAME)
1
2
2
执行后,T1表数据如下(错误!!!):
| FNAME | FMONEY |
|---|---|
| A | 100 |
| B | 空了!!!! |
正确写法:
UPDATE T1
SET T1.FMONEY = (select T2.FMONEY from t2 where T2.FNAME = T1.FNAME)
WHERE EXISTS(SELECT 1 FROM T2 WHERE T2.FNAME = T1.FNAME);
1
2
3
2
3
执行后,T1表数据如下(正确):
| FNAME | FMONEY |
|---|---|
| A | 100 |
| B | 30 |
注意: 必须加最后一行
WHERE EXISTS(SELECT 1 FROM T2 WHERE T2.FNAME = T1.FNAME)判断 避免将 t1 不存在 t2 的字段更新为空值:上述测试数据 t1.B 就被更新为空了,这是不对的。“exists(xxx)”就表示括号里的语句能不能查出记录,它要查的记录是否存在。
因此“select 1”这里的 “1”其实是无关紧要的,换成“*”也没问题,它只在乎括号里的数据能不能查找出来,是否存在这样的记录,如果存在,这 1) 句的where 条件成立。
# 方式2:内联视图更新
UPDATE (
select t1.fmoney fmoney1,t2.fmoney fmoney2 from t1,t2 where t1.fname = t2.fname
)t
set fmoney1 =fmoney2;
1
2
3
4
2
3
4
# 方式3:merge更新
merge into t1
using (select t2.fname,t2.fmoney from t2) t
on (t.fname = t1.fname)
when matched then
update set t1.fmoney = t.fmoney;
1
2
3
4
5
2
3
4
5
# Oracle误修改操作
Oracle提供了另1个快速数据库恢复机制, 就是Flashback 一般只适用于短时间内的恢复, 对于一段相当时间前的误操作, 很可能因为undo数据被覆盖而恢复失败.
Flashback table 操作会修改表里的数据, 从而有可能引起data rows的行移动,那么在执行Flashback table 前必须启用数据行的移动特性.
Alter table <table_name> enable row movement;
--实例 启用row movement属性
alter table t1 enable row movement;
--实例 查看T1,T2是否启用row movement属性
select table_name, row_movement from user_tables where table_name in ('T1','T2');
1
2
3
4
5
2
3
4
5
使用闪回技术恢复数据
--语法
Flashback table <table_name> to SCN <scn_number> [<ENABLE|DISABLE> TRIGGERS];
Flashback table <table_name> to Timestamp <scn_number> [<ENABLE|DISABLE> TRIGGERS];
--实例1 恢复到某个时间点时候
flashback table t1 to timestamp to_timestamp('2021-04-08 12:00:00','yyyy-mm-dd hh24:mi:ss');
--实例2 通过scn恢复
SELECT timestamp_to_scn(to_timestamp('2021-04-08 15:00:00','yyyy-mm-dd hh24:mi:ss')) SCN FROM <table_name>;
FLASHBACK TABLE <table_name> TO SCN 123579067;
1
2
3
4
5
6
7
8
9
10
11
12
13
2
3
4
5
6
7
8
9
10
11
12
13