在Oracle数据库中,如何查找,定位一张表最后一次的DML操作的时间呢? 方式有三种,不过都有一些局限性,下面简单的解析、总结一下。

1:使用ORA_ROWSCN伪列获取表最后的DML时间

ORA_ROWSCN伪列是Oracle 10g开始引入的,可以查询表中记录最后变更的SCN。然后通过SCN_TO_TIMESTAMP函数可以将SCN转换为时间戳,从而找到最后DML操作时SCN的对应时间。但是,默认情况下,每行记录的ORA_ROWSCN是基于Block的,除非在建表的时候开启行级跟踪。

SELECT MAX(ORA_ROWSCN), SCN_TO_TIMESTAMP(MAX(ORA_ROWSCN)) FROM xxx.xxx;

如下所示,我们可以创建一个表TEST,然后查一查TEST表最后的DML的操作时间。如下所示:

  1. SQL>  CREATE TABLE TEST.TEST ( ID NUMBER);

  1.  

  1. Table created.

  1.  

  1. SQL> COL OWNER FOR A12;

  1. SQL> COL TABLE_NAME FOR A32;

  1. SQL> COL MONITORING FOR A32;

  1. SQL> SELECT OWNER, TABLE_NAME, MONITORING

  1.   2  FROM DBA_TABLES

  1.   3  WHERE OWNER='TEST' 

  1.   4    AND TABLE_NAME='TEST';

  1.  

  1. OWNER        TABLE_NAME                       MONITORING

  1. ------------ -------------------------------- --------------------------------

  1. TEST         TEST                             YES

  1.  

  1. SQL> INSERT INTO TEST.TEST VALUES(1);

  1.  

  1. 1 row created.

  1.  

  1. SQL> COMMIT;

  1.  

  1. Commit complete.

  1.  

  1. SQL> SELECT sysdate FROM DUAL;

  1.  

  1. SYSDATE

  1. -------------------

  1. 2018-11-19 14:34:12

  1.  

  1. SQL> SELECT MAX(ORA_ROWSCN), SCN_TO_TIMESTAMP(MAX(ORA_ROWSCN)) FROM TEST.TEST;

  1.  

  1. MAX(ORA_ROWSCN) SCN_TO_TIMESTAMP(MAX(ORA_ROWSCN))

  1. --------------- --------------------------------------------------------------

  1.        52782810 19-NOV-18 02.34.03.000000000 PM

  1.  

  1. SQL>

使用ORA_ROWSCN伪列获取表最新的DML时间,也有一些不足和缺陷,具体如下所示:

1:使用SCN_TO_TIMESTAMP(MAX(ORA_ROWSCN))获取表最后的DML操作时,有可能会遇到ORA-08181错误。

$ oerr ora 8181

08181, 00000, "specified number is not a valid system change number"

// *Cause: supplied scn was beyond the bounds of a valid scn.

// *Action: use a valid scn.

SCN和时间戳的这种转换要依赖于数据库内部的数据记录,而这些数据记录就来自SMON_SCN_TIME基表,具体来说,SMON_SCN_TIME基表用于记录过去时间段中SCN(system change number)与具体的时间戳(timestamp)之间的映射关系,因为是采样记录这种映射关系,所以SMON_SCN_TIME可以较为粗糙地(不精确地)定位某个SCN的时间信息。实际的SMON_SCN_TIME是一张簇表。而且从10g开始SMON也会定期清理SMON_SCN_TIME中的记录,所以对于比较久远的SCN则不能转换。也就出现了数据库某些表使用SCN_TO_TIMESTAMP函数时,会遇到ORA-08181错误,如下所示,我们用比基表SMON_SCN_TIME中MIN(SCN)的还小1的SCN做转换时,就会遇到ORA-08181这个错误。

根据官方文档来看: SMON进程每5分钟采集一次插入到SMON_SCN_TIME表中,同时也删除一些历史数据(超过5天前数据)

This is expected behavior as the SCN must be no older than 5 days as part of the current flashback database

features.

Currently, the flashback query feature keeps track of times up to a

maximum of 5 days. This period reflects server uptime, not wall-clock

time. You must record the SCN yourself at the time of interest, such as

before doing a DELETE.

2: 使用ORA_ROWSCN伪列获取表中某一行的DML操作时间可能不准确,当然对于获取表最后的DML时间是准确的。

默认情况下,每行记录的ORA_ROWSCN是基于数据块(block)的,这样对于某一行最后的DML时间是不准确的,除非在建表的时候执行开启行级跟踪(create table … rowdependencies),这样才会是在行级记录级别的SCN。而每个数据块(block)在头部是记录了该数据块(block)最近事务的SCN,所以默认情况下,只需要从块的头部直接获取这个值就可以了,不需要其他任何的开销。但是这明显是不精确的,一个数据块(block)中会有很多行记录,每次事务不可能影响到整个数据块(block)中所有的行,所以这是一个非常不精准的估算值,同一个数据块(block)的所有记录的ORA_ROWSCN都会是相同的.如下实验所示, 当然对于获取表最后的DML时间是准确的。所以对于每一行的ORA_ROWSCN要求精确的话,就必须开启行级跟踪。

  1. SQL> SELECT * FROM TEST.TEST;

  1.  

  1.         ID

  1. ----------

  1.          1

  1.  

  1. SQL> SELECT ID, SCN_TO_TIMESTAMP(ORA_ROWSCN) FROM TEST.TEST;

  1.  

  1.         ID SCN_TO_TIMESTAMP(ORA_ROWSCN)

  1. ---------- -------------------------------------------------------------------

  1.          1 19-NOV-18 02.34.03.000000000 PM

  1.  

  1. SQL> INSERT INTO TEST.TEST VALUES(2);

  1.  

  1. 1 row created.

  1.  

  1. SQL> COMMIT;

  1.  

  1. Commit complete.

  1.  

  1. SQL> INSERT INTO TEST.TEST VALUES(3);

  1.  

  1. 1 row created.

  1.  

  1. SQL> COMMIT;

  1.  

  1. Commit complete.

  1.  

  1. SQL> SELECT ID, SCN_TO_TIMESTAMP(ORA_ROWSCN) FROM TEST.TEST;

  1.  

  1.         ID SCN_TO_TIMESTAMP(ORA_ROWSCN)

  1. ---------- ---------------------------------------------------------------

  1.          1 19-NOV-18 03.41.01.000000000 PM

  1.          2 19-NOV-18 03.41.01.000000000 PM

  1.          3 19-NOV-18 03.41.01.000000000 PM

3:假如表的数据被TRUNCATE掉或全部DELETE后,也会导致无法定位最后一次DML操作的时间。如下所示:

2:使用DBA_TAB_MODIFICATIONS来查找、定为最后的DML操作时间

 

DBA_TAB_MODIFICATIONS describes modifications to all tables in the database that have been modified since the last time statistics were gathered on the tables

This view is populated only for tables with the MONITORING attribute. It is intended for statistics collection over a long period of time. For performance reasons, the Oracle Database does not populate this view immediately when the actual modifications occur. Run the FLUSH_DATABASE_MONITORING_INFO procedure in the DIMS_STATS PL/SQL package to populate this view with the latest information. The ANALYZE_ANY system privilege is required to run this procedure.

使用DBA_TAB_MODIFICATIONS来查看表最后DML的操作时间,如下测试所示

  1. SQL> CREATE TABLE TEST.TEST (ID  NUMBER);

  1.  

  1. Table created.

  1.  

  1. SQL> COL OWNER FOR A12;

  1. SQL> COL TABLE_NAME FOR A32;

  1. SQL> COL MONITORING FOR A32;

  1. SQL> SELECT OWNER, TABLE_NAME, MONITORING

  1.   2  FROM DBA_TABLES

  1.   3  WHERE OWNER='TEST' 

  1.   4    AND TABLE_NAME='TEST';

  1.  

  1. OWNER        TABLE_NAME                       MONITORING

  1. ------------ -------------------------------- --------------------------------

  1. TEST         TEST                             YES

  1.  

  1. SQL> INSERT INTO TEST.TEST VALUES(1);

  1.  

  1. 1 row created.

  1.  

  1. SQL> COMMIT;

  1.  

  1. Commit complete.

  1.  

  1.  

  1. SQL> ALTER SESSION SET NLS_DATE_FORMAT="YYYY-MM-DD HH24:MI:SS";

  1.  

  1. Session altered.

  1.  

  1. SQL> SELECT INSERTS,UPDATES,DELETES,TRUNCATED,TIMESTAMP 

  1.   2  FROM DBA_TAB_MODIFICATIONS

  1.   3  WHERE TABLE_NAME='TEST' AND TABLE_OWNER='TEST';

  1.  

  1. no rows selected

  1.  

  1. SQL>  EXEC DBMS_STATS.FLUSH_DATABASE_MONITORING_INFO;

  1.  

  1. PL/SQL procedure successfully completed.

  1.  

  1. SQL> SELECT INSERTS,UPDATES,DELETES,TRUNCATED,TIMESTAMP 

  1.   2  FROM DBA_TAB_MODIFICATIONS

  1.   3  WHERE TABLE_NAME='TEST' AND TABLE_OWNER='TEST';

  1.  

  1.    INSERTS    UPDATES    DELETES TRU TIMESTAMP

  1. ---------- ---------- ---------- --- -------------------

  1.          1          0          0 NO  2018-11-20 10:34:24

但是用DBA_TAB_MODIFICATIONS来定位表最后的DML操作时间也有一定的局限性。如下所示,有些局限性会影响定位最后DML操作的时间的准确性。

1:如果表没有设置MONITORING属性,那么DBA_TAB_MODIFICATIONS视图是不会收集相关表的数据的呢。 假如某张表之前没有设置MONITORING属性,那么无法查找最后一次DML操作的时间,设置MONITORING属性后,DBA_TAB_MODIFICATIONS视图里面收集的是这个设置时间点后面的DML操作时间。

2:需要执行EXEC DBMS_STATS.FLUSH_DATABASE_MONITORING_INFO后,视图才会有数据。

3:DML操作不提交或回滚,也会记录到视图中。这样就会导致数据不准确。

未提交情况:

回滚情况:

3:收集完统计信息(ANALYZE或dbms_stats包收集统计信息)后,视图中相关表记录会置空

  1. SQL> SELECT INSERTS,UPDATES,DELETES,TRUNCATED,TIMESTAMP 

  1.   2  FROM DBA_TAB_MODIFICATIONS

  1.   3  WHERE TABLE_NAME='TEST' AND TABLE_OWNER='TEST';

  1.  

  1.    INSERTS    UPDATES    DELETES TRU TIMESTAMP

  1. ---------- ---------- ---------- --- -------------------

  1.          6          0          4 YES 2018-11-20 13:14:08

  1.  

  1. SQL> exec dbms_stats.gather_table_stats('TEST','TEST');

  1.  

  1. PL/SQL procedure successfully completed.

  1.  

  1. SQL> SELECT INSERTS,UPDATES,DELETES,TRUNCATED,TIMESTAMP 

  1.   2  FROM DBA_TAB_MODIFICATIONS

  1.   3  WHERE TABLE_NAME='TEST' AND TABLE_OWNER='TEST';

  1.  

  1. no rows selected

  1.  

  1. SQL>

4:CTAS建立的插入信息不会记录。如下测试所示:

  1. SQL> CREATE TABLE TEST.TEST1

  1.   2  AS

  1.   3  SELECT * FROM TEST.TEST;

  1.  

  1. Table created.

  1.  

  1. SQL> exec dbms_stats.flush_database_monitoring_info;

  1.  

  1. PL/SQL procedure successfully completed.

  1.  

  1. SQL> SELECT INSERTS,UPDATES,DELETES,TRUNCATED,TIMESTAMP 

  1.   2  FROM DBA_TAB_MODIFICATIONS

  1.   3  WHERE TABLE_NAME='TEST1' AND TABLE_OWNER='TEST';

  1.  

  1. no rows selected

5:DBMS_STATS.FLUSH_DATABASE_MONITORING_INFO收集数据会有几秒的延时,这个时间只能接近最后DML时间,而不是精准的。

  1. SQL> COL OWNER FOR A12;

  1. SQL> COL TABLE_NAME FOR A32;

  1. SQL> COL MONITORING FOR A32;

  1. SQL> SELECT OWNER, TABLE_NAME, MONITORING

  1.   2  FROM DBA_TABLES

  1.   3  WHERE OWNER='TEST' 

  1.   4    AND TABLE_NAME='TEST1';

  1.  

  1. OWNER        TABLE_NAME                       MONITORING

  1. ------------ -------------------------------- --------------------------------

  1. TEST         TEST1                            YES

  1.  

  1. SQL>

  1.  

  1. SQL> SELECT SYSDATE FROM DUAL;

  1.  

  1. SYSDATE

  1. -------------------

  1. 2018-11-20 10:46:39

  1.  

  1. SQL> INSERT INTO TEST.TEST VALUES(10);

  1.  

  1. 1 row created.

  1.  

  1. SQL> SELECT SYSDATE FROM DUAL;

  1.  

  1. SYSDATE

  1. -------------------

  1. 2018-11-20 10:46:57

  1.  

  1. SQL> COMMIT;

  1.  

  1. Commit complete.

  1.  

  1. SQL> SELECT SYSDATE FROM DUAL;

  1.  

  1. SYSDATE

  1. -------------------

  1. 2018-11-20 10:47:07

  1.  

  1. SQL> exec dbms_stats.flush_database_monitoring_info;

  1.  

  1. PL/SQL procedure successfully completed.

  1.  

  1. SQL> SELECT INSERTS,UPDATES,DELETES,TRUNCATED,TIMESTAMP 

  1.   2  FROM DBA_TAB_MODIFICATIONS

  1.   3  WHERE TABLE_NAME='TEST' AND TABLE_OWNER='TEST';

  1.  

  1.    INSERTS    UPDATES    DELETES TRU TIMESTAMP

  1. ---------- ---------- ---------- --- -------------------

  1.          3          0          0 NO  2018-11-20 10:47:13

 

 

3:触发器捕获最后DML操作时间

 

使用触发器捕获DML操作的最后时间是最准确的,但是也是性能开销最大的,不推荐使用。

参考资料:

https://docs.oracle.com/cd/B19306_01/server.102/b14237/statviews_2097.htm#i1591024

https://blog.csdn.net/wenzhongyan/article/details/51829271

https://asktom.oracle.com/pls/asktom/f?p=100:11:0::::P11_QUESTION_ID:1260600445561

ORACLE中如何查找定位表最后DML操作的时间小结的更多相关文章

  1. 查看Oracle中是否有锁表

    转: 查看Oracle中是否有锁表 2018-04-23 17:59 alapha 阅读(19450) 评论(0) 编辑 收藏 一.用dba用户登录,或者将用户赋权为DBA用户 命令: su - or ...

  2. Oracle中查看所有的表,用户表,列名,主键,外键

    在Oracle中查看所有的表: select * from tab/dba_tables/dba_objects/cat; 看用户建立的表 : select table_name from user_ ...

  3. 如何恢复oracle中已删除的表

    在9i中Oracle引入了flashback的概念,可以将数据返回到某个时间点,但对于诸如drop/truncate等DDL语句却尚不支持.进入Oracle10g,这一缺陷得到了弥补.可以将丢失掉的表 ...

  4. WindowsPE权威指南-PE文件头中的重定位表

    PE加载的过程 任何一个EXE程序会被分配4GB的内存空间,用户层处理低2G的内存,驱动处理高2G的内存. 1.双击EXE程序,操作系统开辟一个4GB的空间. 2.从ImageBase决定了加载后的基 ...

  5. Oracle中如何查询一个表的所有字段名和数据类型

    Oracle中如何查询一个表的所有字段名和数据类型 查询语法 select A.COLUMN_NAME,A.DATA_TYPE from user_tab_columns A where TABLE_ ...

  6. Oracle中把一张表查询结果插入到另一张表中

      1. 新增一个表,通过另一个表的结构和数据 create table XTHAME.tab1 as select * from DSKNOW.COMBDVERSION 2. 如果表存在: inse ...

  7. oracle中,约束、表名、Index等的名称长度限制最大只能30个字符

    oracle中,约束.表名.Index等的名称长度限制最大只能30个字符

  8. 创建日志表记录DML操作和DDL操作

    创建一个日志表,记录dept表的DML操作 create table dept_log(logid number,type varchar2(50),logdate date,deptno numbe ...

  9. oracle中查询某张表都被哪些表参照了

    起因: 系统测试的时候发现如果某条记录已经被引用了,这个时候删除这条记录会引起数据不一致,系统会报错.比如警员信息,在考勤记录表里会引用警员ID,如果考勤记录表中已经存在这个警员ID了,这时从警员表中 ...

随机推荐

  1. [Swift]LeetCode775. 全局倒置与局部倒置 | Global and Local Inversions

    We have some permutation Aof [0, 1, ..., N - 1], where N is the length of A. The number of (global) ...

  2. [Swift]LeetCode907. 子数组的最小值之和 | Sum of Subarray Minimums

    Given an array of integers A, find the sum of min(B), where B ranges over every (contiguous) subarra ...

  3. ubuntu---网络管理

    网络配置文件 在 /etc/network/interface auto eth0 iface eth0 inet static address x.x.x.x netmask 255.255.255 ...

  4. Spring Data Jpa接口简介

    Repository接口 public interface Repository<T, ID> {....} 提供了按方法名称的查询方式: 提供了@Query的查询方式 可能遇到的错误: ...

  5. B+树的Copy-on-Write设计

    本文主要介绍B+树的Copy-On-Write,包括由来.设计思路和核心源码实现(以Xapian源码为例).中文的互联网世界里,对B树.B+树的科普介绍很丰富,但对它们在工业界的实际使用却几乎没有相关 ...

  6. arrays.xml中使用integer-array引用drawable图片资源,代码中如何将这些图片资源赋值到ImageView控件中

    当我们在arrays.xml文件中声明一些图片资源数组的时候: <?xml version="1.0" encoding="utf-8"?> < ...

  7. [Leetcode]674. Longest Continuous Increasing Subsequence

    Given an unsorted array of integers, find the length of longest continuous increasing subsequence. E ...

  8. (一)你的第一个Socket程序

    概述 本文通过一个最简单的Socket通信来对每一步做通俗易懂的讲解让你了解这些函数到底是干什么用的.下面的代码虽然是用Pyhton实现的,但是你要知道这些通信机制并不是Python所定义的,因为这些 ...

  9. 【ASP.NET Core快速入门】(四)在CentOS上安装.NET Core运行时、部署到CentOS

    下载.NET Core SDK 下载地址:https://www.microsoft.com/net/download/windows 第一步:Add the dotnet product feed( ...

  10. [二]Java虚拟机 jvm内存结构 运行时数据内存 class文件与jvm内存结构的映射 jvm数据类型 虚拟机栈 方法区 堆 含义

    前言简介 class文件是源代码经过编译后的一种平台中立的格式 里面包含了虚拟机运行所需要的所有信息,相当于 JVM的机器语言 JVM全称是Java Virtual Machine  ,既然是虚拟机, ...