在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时间。如下所示:

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

Table created.

SQL> COL OWNER FOR A12;SQL> COL TABLE_NAME FOR A32;SQL> COL MONITORING FOR A32;SQL> SELECT OWNER, TABLE_NAME, MONITORING   2  FROM  DBA_TABLES   3  WHERE OWNER='TEST'   4    AND TABLE_NAME='TEST';

OWNER        TABLE_NAME                       MONITORING------------ -------------------------------- --------------------------------TEST         TEST                             YES

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

1 row created.

SQL> COMMIT;

Commit complete.

SQL> SELECT sysdate FROM DUAL;

SYSDATE-------------------2018-11-19 14:34:12

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

MAX(ORA_ROWSCN) SCN_TO_TIMESTAMP(MAX(ORA_ROWSCN))--------------- --------------------------------------------------------------       52782810 19-NOV-18 02.34.03.000000000 PM

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要求精确的话,就必须开启行级跟踪。

SQL> SELECT * FROM TEST.TEST;

        ID----------         1

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

        ID SCN_TO_TIMESTAMP(ORA_ROWSCN)---------- ---------------------------------------------------------------------------         1 19-NOV-18 02.34.03.000000000 PM

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

1 row created.

SQL> COMMIT;

Commit complete.

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

1 row created.

SQL> COMMIT;

Commit complete.

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

        ID SCN_TO_TIMESTAMP(ORA_ROWSCN)---------- ---------------------------------------------------------------------------         1 19-NOV-18 03.41.01.000000000 PM         2 19-NOV-18 03.41.01.000000000 PM         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的操作时间,如下测试所示

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

Table created.

SQL> COL OWNER FOR A12;SQL> COL TABLE_NAME FOR A32;SQL> COL MONITORING FOR A32;SQL> SELECT OWNER, TABLE_NAME, MONITORING   2  FROM DBA_TABLES   3  WHERE OWNER='TEST'   4    AND TABLE_NAME='TEST';

OWNER        TABLE_NAME                       MONITORING------------ -------------------------------- --------------------------------TEST         TEST                             YES

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

1 row created.

SQL> COMMIT;

Commit complete.

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

Session altered.

SQL> SELECT INSERTS,UPDATES,DELETES,TRUNCATED,TIMESTAMP   2  FROM DBA_TAB_MODIFICATIONS   3  WHERE TABLE_NAME='TEST' AND TABLE_OWNER='TEST';

no rows selected

SQL>  EXEC DBMS_STATS.FLUSH_DATABASE_MONITORING_INFO;

PL/SQL procedure successfully completed.

SQL> SELECT INSERTS,UPDATES,DELETES,TRUNCATED,TIMESTAMP   2  FROM DBA_TAB_MODIFICATIONS   3  WHERE TABLE_NAME='TEST' AND TABLE_OWNER='TEST';

   INSERTS    UPDATES    DELETES TRU TIMESTAMP---------- ---------- ---------- --- -------------------         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包收集统计信息),视图中相关表记录置空
SQL> SELECT INSERTS,UPDATES,DELETES,TRUNCATED,TIMESTAMP   2  FROM DBA_TAB_MODIFICATIONS   3  WHERE TABLE_NAME='TEST' AND TABLE_OWNER='TEST';

   INSERTS    UPDATES    DELETES TRU TIMESTAMP---------- ---------- ---------- --- -------------------         6          0          4 YES 2018-11-20 13:14:08

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

PL/SQL procedure successfully completed.

SQL> SELECT INSERTS,UPDATES,DELETES,TRUNCATED,TIMESTAMP   2  FROM DBA_TAB_MODIFICATIONS   3  WHERE TABLE_NAME='TEST' AND TABLE_OWNER='TEST';

no rows selected

SQL> 
  • 4:CTAS建立的插入信息不会记录。如下测试所示:
SQL> CREATE TABLE TEST.TEST1  2  AS  3  SELECT * FROM TEST.TEST;

Table created.

SQL> exec dbms_stats.flush_database_monitoring_info;

PL/SQL procedure successfully completed.

SQL> SELECT INSERTS,UPDATES,DELETES,TRUNCATED,TIMESTAMP   2  FROM DBA_TAB_MODIFICATIONS   3  WHERE TABLE_NAME='TEST1' AND TABLE_OWNER='TEST';

no rows selected
  • 5:DBMS_STATS.FLUSH_DATABASE_MONITORING_INFO收集数据会有几秒的延时,这个时间只能接近最后DML时间,而不是精准的。
SQL> COL OWNER FOR A12;SQL> COL TABLE_NAME FOR A32;SQL> COL MONITORING FOR A32;SQL> SELECT OWNER, TABLE_NAME, MONITORING   2  FROM DBA_TABLES   3  WHERE OWNER='TEST'   4    AND TABLE_NAME='TEST1';

OWNER        TABLE_NAME                       MONITORING------------ -------------------------------- --------------------------------TEST         TEST1                            YES

SQL> 

SQL> SELECT SYSDATE FROM DUAL;

SYSDATE-------------------2018-11-20 10:46:39

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

1 row created.

SQL> SELECT SYSDATE FROM DUAL;

SYSDATE-------------------2018-11-20 10:46:57

SQL> COMMIT;

Commit complete.

SQL> SELECT SYSDATE FROM DUAL;

SYSDATE-------------------2018-11-20 10:47:07

SQL> exec dbms_stats.flush_database_monitoring_info;

PL/SQL procedure successfully completed.

SQL> SELECT INSERTS,UPDATES,DELETES,TRUNCATED,TIMESTAMP   2  FROM DBA_TAB_MODIFICATIONS   3  WHERE TABLE_NAME='TEST' AND TABLE_OWNER='TEST';

   INSERTS    UPDATES    DELETES TRU TIMESTAMP---------- ---------- ---------- --- -------------------         3          0          0 NO  2018-11-20 10:47:13

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

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

ora 所有表记录数_ORACLE中如何查找定位表最后DML操作的时间小结相关推荐

  1. mysql查看当前库所有表记录数_mysql中查看数据库中所有表的记录数

    我们继续我们的博客创建.           在一个项目日常的运行中,常常会为服务过慢而导致用户体验不好,在这一点上,常常需要进行性能优化,而如果没有针对性的对某一个 模块进行优化,那么效果往往是不理 ...

  2. java 获取mysql表的大小_oracle中查询表大小和表空间大小 JAVA

    1.查看Oracle数据库中表空间信息的工具方法: 使用oracle enterprise manager console工具,这是oracle的客户端工具,当安装oracle服务器或客户端时会自动安 ...

  3. Oracle查询库中记录数大于2千万的所有表

    Oracle查询库中记录数大于2千万的所有表 假如当前用户拥有select any table权限,则可以使用下列sql语句: select table_name, num_rows from dba ...

  4. MySQL记录变化工具_MySQL 表记录数对比工具

    工具介绍: 以下介绍一个实时对比两个MySQL实例的中表记录数的工具 checkcount.py . 工具大多用于迁移后或者主从数据快速比较 , 快速验证两个mysql数据库实例每个表是否具有相同记录 ...

  5. Oracle 记录表名和表记录数

    oracle那个系统表同时记录表名和表记录数的? select table_name , num_rows from user_tables;

  6. MySQL查询数据库表记录数

    mysql统计一个数据库里所有表的数据量,最近在做统计想查找一个数据库里基本所有的表数据量,数据量少的通过select count再加起来也是可以的,不过表的数据有点多,不可能一个一个地查.记得在Na ...

  7. mysql查询表的列名主键_Oracle中查看所有的表,用户表,列名,主键,外键

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

  8. 达梦数据库统计模式下所有表记录数

    /* 统计每个表的数据量到表数据记录表 */ --创建一张表记录统计信息 create table test(tab_name varchar(30),row_num int) --统计每个表的数据量 ...

  9. oracle 其他用户表主键,Oracle中查看所有的表,用户表,列名,主键,外键

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

最新文章

  1. B - The Suspects POJ - 1611
  2. 运维老鸟告诉你这个经典Zookeeper问题的根因
  3. ncnn windows
  4. spring boot之eureka-client的pom
  5. synchronized的可重入怎么实现的
  6. 百度地图API的使用教程以及案例
  7. python入门--字典的特点
  8. Atitit..文件上传组件选型and最佳实践总结(3)----断点续传控件的实现
  9. 最新Unity 3D游戏开发学习资料集合
  10. 天语手机android 4.4.4,天语 touch 2 4.4.4 ROM刷机包 FIUI
  11. 2022.4.10 ShanDong Multi-University Training #1 QLU-team#8 の 赛后总结
  12. DOM4j解析XMl中碰到的问题解决
  13. CSS实现进度条和订单进度条
  14. 实用指南:如何在Anolis OS上轻松使用 Kata 安全容器?
  15. Unreal Engin_画廊制作笔记 _005灯光处理,平行光的设置
  16. 数据库中case when 的用法
  17. 申请【华为·云享专家】流程
  18. 利用NAS免费部署动态解析实现内网穿透
  19. 蓝牙脂肪秤模块测量原理
  20. 电源去耦的原因-和如何电源去耦

热门文章

  1. 开发ffmpeg/live555常见问题错误及解决方法
  2. 【Ubuntu16.04】解决Qt安装包(.run文件)不能用./命令执行的问题
  3. 前端常见跨域解决方案
  4. 为什么ConcurrentHashMap没有ConcurrentHashSet
  5. Docker如何更改存储库名称或重命名图像?
  6. JavaScript中的“ new”关键字是什么?
  7. Window+Java环境证书生成说明,pfx证书已经生成
  8. java textvaluechanged 全选删除不触发_具有TextChanged事件的AutoCompleteBox未正确选择
  9. 全向轮移动平台参数校准
  10. 问答 | 为什么两轮差速机器人转向运动是圆周运动