一、多表更新

比如线上有个系统由于某一个模块出现异常,导致系统整体的数据出现了错误,需要你手动改写数据库错误,Oracle update语句更新的值来自另一张表

update语法最基本的格式为

UPDATE TABLE_NAME SET COLUMN1 = VALUE1 WHERE COLUMN2 = VALUE2

下面首先来创建两张表,令表名为GF_CONS_TRAN_TEST  AS A

-- ----------------------------
-- Table structure for GF_CONS_TRAN_TEST
-- ----------------------------
CREATE TABLE GF_CONS_TRAN_TEST (
"ID" VARCHAR2(64 BYTE) NOT NULL ,
"USERNAME" VARCHAR2(255 BYTE) NULL ,
"PASSWORD" VARCHAR2(255 BYTE) NULL ,
"CREATEDATE" DATE NULL ,
"PHONE" VARCHAR2(255 BYTE) NULL
)
LOGGING
NOCOMPRESS
NOCACHE ; -- ----------------------------
-- Records of GF_CONS_TRAN_TEST
-- ----------------------------
INSERT INTO GF_CONS_TRAN_TEST VALUES ('0001', '国网金州新区', '123445', TO_DATE('2019-11-29 10:25:23', 'YYYY-MM-DD HH24:MI:SS'), null);
INSERT INTO GF_CONS_TRAN_TEST VALUES ('0002', '大连供电公司', 'phhs_all', TO_DATE('2019-11-18 10:25:59', 'YYYY-MM-DD HH24:MI:SS'), null);
INSERT INTO GF_CONS_TRAN_TEST VALUES ('0003', '中山供电公司', 'zs_phs', TO_DATE('2019-11-29 10:26:49', 'YYYY-MM-DD HH24:MI:SS'), null); -- ----------------------------
-- Indexes structure for table GF_CONS_TRAN_TEST
-- ---------------------------- -- ----------------------------
-- Checks structure for table GF_CONS_TRAN_TEST
-- ----------------------------
ALTER TABLE GF_CONS_TRAN_TEST ADD CHECK ("ID" IS NOT NULL); -- ----------------------------
-- Primary Key structure for table GF_CONS_TRAN_TEST
-- ----------------------------
ALTER TABLE GF_CONS_TRAN_TEST ADD PRIMARY KEY ("ID");

A表结构及数据如下:

再创建一张和A表结构类似的表B ,GF_CONS_TRAN_TEST_COPY  AS B

-- ----------------------------
-- Table structure for GF_CONS_TRAN_TEST_COPY
-- ----------------------------
CREATE TABLE GF_CONS_TRAN_TEST_COPY (
"ID" VARCHAR2(64 BYTE) NOT NULL ,
"USERNAME" VARCHAR2(255 BYTE) NULL ,
"PASSWORD" VARCHAR2(255 BYTE) NULL ,
"CREATEDATE" DATE NULL
)
LOGGING
NOCOMPRESS
NOCACHE ; -- ----------------------------
-- Records of GF_CONS_TRAN_TEST_COPY
-- ----------------------------
INSERT INTO GF_CONS_TRAN_TEST_COPY VALUES ('0001', '国网金州新区_更新', '123445', null);
INSERT INTO GF_CONS_TRAN_TEST_COPY VALUES ('0003', '中山供电公司_更新', 'zs_phs', null); -- ----------------------------
-- Indexes structure for table GF_CONS_TRAN_TEST_COPY
-- ---------------------------- -- ----------------------------
-- Checks structure for table GF_CONS_TRAN_TEST_COPY
-- ----------------------------
ALTER TABLE "USER_PSSC"."GF_CONS_TRAN_TEST_COPY" ADD CHECK ("ID" IS NOT NULL); -- ----------------------------
-- Primary Key structure for table GF_CONS_TRAN_TEST_COPY
-- ----------------------------
ALTER TABLE GF_CONS_TRAN_TEST_COPY ADD PRIMARY KEY ("ID");

B表结构及数据如下

可以看到第一张表(GF_CONS_TRAN_TEST AS A)中有三条数据,第二张表(GF_CONS_TRAN_TEST_COPY AS B)有两条数据,都有相同的字段ID,(要不然没有更新条件),假设这里需要更新A表中的USENAME列为表B中的数据,因为有唯一主键ID做索引,我们可以这样写

UPDATE GF_CONS_TRAN_TEST A SET A.USERNAME = (
SELECT B.USERNAME FROM GF_CONS_TRAN_TEST_COPY B WHERE A.ID = B.ID
)

结果显示更新成功,但是受影响的行是3行,我们再看下表A更新后数据

由于表B中没有记录0002,因为用刚才的语句 WHERE A.ID = B.ID所以将表A的数据全部给更新了,表B中没有的显示为空了,所以说把之前的数据整没了,但是老板可没这么说啊,

幸好,Oracle数据库在我们执行插入、更新、删除等操作时,容易产生误操作,导致数据库其他的内容被修改,通过普通的sql语句操作无法还原,可采用Oracle数据库表的闪回机制,根据时间戳,

二、还原表

1.先查询这个时间点的数据是否为要还原的数据

SELECT
*
FROM
TABLENAME AS OF TIMESTAMP TO_TIMESTAMP (
'2019-07-05 08:50:00',
'YYYY-MM-DD HH24:MI:SS'
) --表某个时间段的数据

可以看到数据为我们更新之前的原始数据

2.开启这张表的闪回机制

ALTER TABLE TABLE_NAME ENABLE ROW MOVEMENT

3.把表的状态闪回到这个时间段

FLASHBACK TABLE TABLE_NAME  TO TIMESTAMP TO_TIMESTAMP('2019-11-29 11:00:00','YYYY-MM-DD HH24:MI:SS')

4.关闭这张表的闪回状态(切勿忘记执行该步骤)

ALTER TABLE TABLE_NAME DISABLE ROW MOVEMENT

执行完毕会发现表A的数据已经还原到这个时间之前的状态了,因为之前写的update语句会更新全部的记录,所以我们加入限制条件,当B表的记录主键ID在表A中存在时开始更新,SQL语句如下:

UPDATE GF_CONS_TRAN_TEST A SET A.USERNAME = (SELECT B.USERNAME FROM GF_CONS_TRAN_TEST_COPY B WHERE A.ID = B.ID)
WHERE EXISTS (SELECT 1 FROM GF_CONS_TRAN_TEST_COPY B WHERE B.ID = A.ID)

我们可以看到这回更新了两条记录,与我们所想完全一致,回头查看表A的数据

这两条数据为我们刚才所更新的数据,并且0002数据没有变化,符合当初的要求

三、MERGE命令

在oracle9g引入了MERGE命令,可以在一个sql语句中对一个表同时进行insert和update操作,merge命令可以从一个或多个数据源中选择行来update或者insert到一个表或多个表。

Merge into的作用就是解决B表跟新A表数据,如果A表没有,则把B表数据插入A表或向一个表中插入数据,如果该表有该数据则更新,如果没有则插入。

一般我们执行插入语句会是这样

INSERT INTO GF_CONS_TRAN_TEST (ID, USERNAME, PASSWORD) VALUES('0004','国网庄河供电','zh_org')

当然对于主键ID来说常用的两种方式就是UUID和自增长了,很少会出现主键重复的问题,但是假如在插入数据的时候根据主键4判断,如果没有插入数据,如果有则根据ID来更新,岂不一举两得

为了可以看到效果,我们手动更新A表数据如下

B表数据不变

B表中有0001在A表中没有,0003在A表中存在,0002在B表中没有,如果没有出错应该是更新0003,插入0001,SQL执行如下:

MERGE INTO GF_CONS_TRAN_TEST A USING (SELECT B.ID, B.USERNAME, B.PASSWORD FROM GF_CONS_TRAN_TEST_COPY B ) C ON (A.ID = C.ID)
WHEN MATCHED THEN
UPDATE SET A.USERNAME = C.USERNAME, A.PASSWORD = C.PASSWORD
WHEN NOT MATCHED THEN
INSERT (A.ID, A.USERNAME, A.PASSWORD) VALUES(C.ID, C.USERNAME, C.PASSWORD)

执行结构如下:

与我们所想一致,一条为更新操作,一条为新增操作。

Oracle多表更新及MERGE命令和闪回机制还原数据表的更多相关文章

  1. Oracle 中利用闪回查询确定某表在某时间点之后的修改内容,并恢复至该时间点

    Oracle 中利用闪回查询确定某表在某时间点之后的修改内容: 1.查看 DELETE 及 UPDATE 操作修改的数据: SQL> SELECT * FROM tab AS OF TIMEST ...

  2. Oracle Flashback Technologies - 闪回被drop的表

    Oracle Flashback Technologies - 闪回被drop的表 在oracle10g中,drop一个表后,表没有真正被删除,支持被rename后放在recyclebin中. #新建 ...

  3. Oracle闪回恢复误删除的表、存储过程、包、函数...

    在日常的数据库开发过程汇总难免会出现一些误删除的动作, 对于一些误删的操作我们可以通过oracle提供的闪回机制恢复误删数据, 从而避免出现较大的生产事故. 下面是本人平时工作中积累的一些常用的操作, ...

  4. 【oracle11g,13】表空间管理2:undo表空间管理(调优) ,闪回原理

    一.undo空间原理: dml操作会产生undo数据. update时,sever process 会在databuffer 中找到该记录的buffer块,没有就从datafile中找并读入data ...

  5. SQL Fundamentals: 表的创建和管理(表的基本操作,闪回技术flashback,表结构修改)

    SQL Fundamentals || Oracle SQL语言 1.表的基本操作 (CREATE TABLE, DROP TABLE,TRUNCATE TABLE, RENAME tablename ...

  6. ORACLE闪回机制分析与研究应用

    1.查看数据库归档和闪回状态,及环境准备SQL> archive log list;SQL> select flashback_on from v$database;关闭数据库,启动归档和 ...

  7. Oracle的闪回技术--闪回已删除的表

    注意闪回技术只能保护非系统表决空间中的表,而且表空间必须本地管理, 外键将不可以被恢复, 索引和约束的名字将会被命名为以BIN开头,由系统生成的名字 查看是否开启闪回: SQL> show pa ...

  8. 【Oracle】闪回drop后的表

    本文介绍的闪回方式只适用于:删除表的表空间非system,drop语句中没有purge关键字(以上两种情况的误删除操作只能通过日志找回): 1.删除表后直接从回收站闪回 SCOTT@LGR> d ...

  9. 己亥清爽恢复系列之数据文件4篇:DROP表后如何恢复(非闪回技术)

    己亥清爽系列说明:清爽系列是作为恢复系列的基础篇,基于FS(File System)文件系统的手工还原恢复,也叫基于用户管理的还原恢复,来自于博客园AskScuti. 实验说明:你不小心Drop掉了一 ...

随机推荐

  1. C++中static的作用

    tatic关键字有俩作用:(1).控制存储分配:(2).控制一个名字的可见性和连接.   随着C++名字空间的引入,我们有了更好的,更灵活的方法来控制一个大项目的名字增长.     在类的内部使用st ...

  2. java中wait()和sleep()的区别

    前言 以前只知道一个结论,但是没法理解,现在水平上来了,自己代码中写了一个验证的方法. 1.先上结论:wait()会释放持有的锁,sleep()不会释放持有的锁 2.验证:看代码运行结果. packa ...

  3. word--->pdf资料转载..

    https://blog.csdn.net/dsn727455218/article/details/80667927

  4. QML学习笔记(八)— QML实现列表侧滑覆盖按钮

    QML实现列表右边滑动删除按钮,并覆盖原有的操作按钮,点击可实现删除当前项 本文链接:QML实现列表侧滑覆盖按钮 作者:狐狸家的鱼 GitHub:八至 列表实现在另一篇博客已经提及,列表可选中.拖拽. ...

  5. python 在图像上写中文字体 (python write Chinese in image)

    本人处理图像的时候经常使用opencv的包,但是 cv2.putText 显示不了中文,所以查找了如何在python在图像上写中文的方法,在伟大的Stack Overflow上面找到一个方法,分享给大 ...

  6. MyISAM和InnoDB引擎的区别

    MySQL默认采用的是MyISAM. MyISAM不支持事务,而InnoDB支持.InnoDB的AUTOCOMMIT默认是打开的,即每条SQL语句会默认被封装成一个事务,自动提交,这样会影响速度,所以 ...

  7. kvm搭建

    今日做公司项目时恰好是这个,然后就做一个博客,希望可以帮助到你们 1.把虚拟机zmedu63内存调成6G以上,因为我们要在VMware虚拟中安装KVM,然后在KVM中再安装虚拟机,需要内存大一些.zm ...

  8. Delphi 获取系统的语言环境参数GetSystemDefaultLangID、VerLanguageName、GetLocaleInfo

    1 核心的两个API函数:GetSystemDefaultLangID 和 VerLanguageName. GetSystemDefaultLangID:获得系统默认语言的ID VerLanguag ...

  9. python基础四(文件操作)

    1.文件操作的基本流程 1.1 文件操作的基本流程 # 绝对路径 # r"d:/葫芦娃.txt 前面的r 为转义. linux平台和windows的分隔符不同 f = open(r" ...

  10. 使用JQuery对页面进行绑值

    使用JQuery对页面进行绑值 <!DOCTYPE html> <html> <head> <meta charset="UTF-8"&g ...