阿里云-云小站(无限量代金券发放中)
【腾讯云】云服务器、云数据库、COS、CDN、短信等热卖云产品特惠抢购

Oracle redo日志产生量测试及比较

208次阅读
没有评论

共计 16697 个字符,预计需要花费 42 分钟才能阅读完成。

Oracle 数据库日志有几种模式,强制日志、非强制日志模式、对象级别等!通常在数据库模式设置强制后,余下的不管如何都会强制日志!现在我们就 oracle 数据库日志模式产生的日志量进行比较测试,在各种模式下日志产生的情况!

日志记录模式的转换
      a. 数据库从非强制日子模式切换到强制日志模式
          SQL>alter database force logging;   
        b. 数据库从强制日志模式切换到非强制日志模式
          SQL>alter database noforce logging;
        c. 表空间级别从强制日志模式切换到非强制日志模式
          SQL>alter tablespace tablespacename noforce logging;
        d. 表空间级别从非强制日志模式切换到强制日志模式
          SQL>alter tablespace tablespacename force logging;
        e. 对象级别日志记录模式
          SQL>alter tablet mytest nologging;    – 不记录日志模式
          SQL>alter tablet mytest logging;      – 采用日志记录模式
 
一、表段,索引段上使用一般 DDL,DML 时,LOGGING 与 NOLOGGING 情况 
    1. 查看数据库的归档模式
        有关设置日志归档模式的问题,请参考:
            Oracle 联机重做日志文件(ONLINE LOG FILE)
            Oracle 归档日志
 
        SQL> select log_mode,force_logging from v$database;
 
        LOG_MODE    FOR
        ———— —
        ARCHIVELOG  NO
       
        SQL> archive log list;
        Database log mode              Archive Mode
        Automatic archival            Enabled
        Archive destination            /u01/mytest/arch
        Oldest online log sequence    1024
        Next log sequence to archive  1025
        Current log sequence          1025 
       
        SQL> select tablespace_name,logging,force_logging from dba_tablespaces;
 
        TABLESPACE_NAME                LOGGING  FOR
        —————————— ——— —
        SYSTEM                        LOGGING  NO
        UNDOTBS1                      LOGGING  NO
        SYSAUX                        LOGGING  NO
        TEMP                          NOLOGGING NO
        USERS                          LOGGING  NO
        PERFSTAT                      LOGGING  NO
       

     
    2. 使用如下语句进行查询
     
          SQL >SELECT a.name,b.value  FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic# WHERE a.name = ‘redo size’;
 

 
    3. 在归档模式下比较表段上的 NOLOGGING 与 LOGGING
        SQL>  SELECT a.name,b.value  FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic# WHERE a.name = ‘redo size’;
 
        NAME                VALUE
        ————— ———-
        redo size            1644
 
        SQL> CREATE TABLE mytest_nolog NOLOGGING AS SELECT * FROM dba_objects;  –nologging 模式创建表
 
        SQL> SELECT a.name,b.value  FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic# WHERE a.name = ‘redo size’;
 
        NAME                VALUE
        ————— ———-
        redo size            70064         
 
        SQL> select 1644 last,70064 as cur,(70064-1644) diff from dual;– 使用 nologging 模式建表产生的 redo size 为
 
              LAST        CUR      DIFF
        ———- ———- ———-
              1644      70064      68420
 
        SQL> CREATE TABLE mytest LOGGING AS SELECT * FROM dba_objects;    – 使用 logging 模式来创建表
 
        SQL> SELECT a.name,b.value  FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic# WHERE a.name = ‘redo size’;;  – 查看当前的 redo size
 
        NAME                VALUE
        ————— ———-           
        redo size          1344112             

        SQL> select 70064 last,1344112 as cur,(1344112-70064) diff from dual; 
 
              LAST        CUR      DIFF
        ———- ———- ———-
            70064    1344112      1274048          —- 查看 logging 模式产生的 redo size 为 -68420=1274048, 比 nologging 日志模, 有 19 倍多!
             
 
        SQL> select table_name,logging from user_tables where table_name like ‘MYTEST%’;– 查看创建表的日志记录模式
 
        TABLE_NAME                    LOG
        —————————— —
        MYTEST                        YES
        MYTEST_NOLOG                  NO
 
    4. 基于索引来比较 redo size(同样是在归档模式下)
        SQL> SELECT a.name,b.value  FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic# WHERE a.name = ‘redo size’;;    – 查看当前的 redo_size
 
        NAME                VALUE
        ————— ———-
        redo size            1140
 
        SQL> create index idx_mytest on mytest(object_id);  – 基于表 mytest 来创建索引
 
        SQL> SELECT a.name,b.value  FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic# WHERE a.name = ‘redo size’;;      – 查看当前的 redo_size
 
        NAME                VALUE
        ————— ———-                – 基于 loggiing 模式,创建索引产生的 redo size 为 221600-1140=220460
        redo size          221600
 
        SQL> alter index idx_mytest rebuild;    – 重建索引
           
        SQL> select 221600 last,448132 cur,448132-221600 diff from dual;
       
              LAST        CUR      DIFF
        ———- ———- ———-
            221600    448132    226532  – 重建索引产生的日志比直接建还要多,主要是重建过程还有一个对旧索引的删除
 
             
        SQL> alter index idx_mytest rebuild nologging; – 使用 nologging 重建索引。
                                                – 也可以在创建索引的时候直接使用 nologging 关键字
                                                – 如:create index idx_mytest_nolog mytest_nolog(object_id) nologging
        SQL> SELECT a.name,b.value  FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic# WHERE a.name = ‘redo size’;;    – 查看当前的 redo size
 
        NAME                VALUE
        ————— ———-              – 基于 nologging 日志模式重建索引产生的 redo size 为 469160-448132=21028
        redo size          469160
 
    5. 非归档模式下的 LOGGING 与 NOLOGGING
 
        SQL> drop table mytest purge;
 
        SQL> drop table mytest_nolog purge;
       
        SQL> select log_mode,force_logging from v$database;  – 切换日志到非归档模式后,下面是查询的结果
 
        LOG_MODE    FOR
        ———— —
        NOARCHIVELOG NO
       
        SQL> SELECT a.name,b.value  FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic# WHERE a.name = ‘redo size’;;    – 查看当前的 redo size
 
        NAME                VALUE
        ————— ———-
        redo size            1688
        SQL> create table mytest as select * from dba_objects;  – 创建表对象,使用 logging 日志记录模式
 
        SQL> SELECT a.name,b.value  FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic# WHERE a.name = ‘redo size’;;  – 查看当前的 redo size
 
        NAME                VALUE
        ————— ———-            – 使用 logging 日志记录模式,创建表对象之后产生的 redo size 为 -1688 =68548
        redo size            70236
 
        SQL> create table mytest_nolog nologging as select * from dba_objects; – 创建表对象,使用 nologging 日志记录模式
 
        SQL> SELECT a.name,b.value  FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic# WHERE a.name = ‘redo size’;;  – 查看当前的 redo size
 
        NAME                VALUE
        ————— ———-            – 使用 nologging 日志记录模式创建表对象之后产生的 redo size 为 135464-70236=65228
 
        redo size          135464
         
    6. 小结:
        使用 logging 与 nologging 来创建对象或执行 DML 时
            对于非归档模式下,其产生的日志信息(redo size) 相差的并不大
            对于归档模式下,logging 模式产生的日志将远远大于使用 nologging 模式产生的日志量

更多详情见请继续阅读下一页的精彩内容:http://www.linuxidc.com/Linux/2016-06/132016p2.htm

现在我们通过普通的 insert,append,以及创建表的方式进行比较
 
    1. 数据库运行在非归档模式下
        a. 使用 logging 模式创建表
            SQL> select log_mode from v$database;
 
            LOG_MODE
            ————
            NOARCHIVELOG 
 
            SQL> SELECT a.name,b.value FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic#  WHERE a.name = ‘redo size’;
 
            NAME                VALUE
            ————— ———-
            redo size            1764
 
            SQL> create table mytest as select * from dba_objects where 1=0;
 
            SQL> SELECT a.name,b.value FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic#  WHERE a.name = ‘redo size’;
 
            NAME                VALUE
            ————— ———-    – 建表产生的 redo 量 23908-1764=22144
            redo size            23908 
 
            SQL> insert into mytest select * from dba_objects;
 
            11634 rows created.
            Elapsed: 00:00:00.36
 
            SQL> SELECT a.name,b.value FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic#  WHERE a.name = ‘redo size’;
 
            NAME                VALUE
            ————— ———-    – 直接使用 insert 时产生的 redo 量 1281060-23908=1257152
            redo size          1281060 
 
            SQL> insert /*+ append */ into mytest select * from dba_objects;
 
            11634 rows created.
            Elapsed: 00:00:00.26
 
            SQL> SELECT a.name,b.value FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic#  WHERE a.name = ‘redo size’;
 
            NAME                VALUE
            ————— ———-    – 使用 append 模式时产生的 redo 量 1284740-1281060=3680
            redo size          1284740    – 普通 insert 比使用 append insert 多产生 1257152/3680=341 倍 redo
 
        b. 使用 nologging 模式创建表
            SQL> create table mytest_nolog nologging as select * from dba_objects where 1=0;
 
            SQL> SELECT a.name,b.value FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic#  WHERE a.name = ‘redo size’;
 
            NAME                VALUE
            ————— ———-    – 使用 nologging 创建空表 mytest_nolog 时产生的日志量 1305812-1284740=21072
            redo size          1305812
 
            SQL> insert into mytest_nolog select * from dba_objects;
 
            11635 rows created.
            Elapsed: 00:00:00.21
            SQL> SELECT a.name,b.value FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic#  WHERE a.name = ‘redo size’;                       
 
            NAME                VALUE
            ————— ———-    – 使用普通 insert 插入记录产生的日志量 2562664-1305812=1256852
            redo size          2562664
 
            SQL> insert /* +append */ into mytest_nolog select * from dba_objects;
 
            11635 rows created.
            Elapsed: 00:00:00.18
            SQL> SELECT a.name,b.value FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic#  WHERE a.name = ‘redo size’;                                   

 
 
            NAME                VALUE
            ————— ———-      – 使用 append 模式时产生的 redo 量 3766404-2562664=1203740
            redo size          3766404   
                                           
        c.redo 的比较
            在具有 logging 属性对象中,使用 append 模式时产生的 redo 量 1284740-1281060=3680,
                普通 insert 比使用 append insert 多产生 /3680=341 倍 redo
            在具有 nologging 属性对象中, 使用 append insert 模式与普通 insert 模式产生的 redo 量相差不太大,
                append insert 模式为,而普通的 insert 模式为
         
    2. 数据库运行在归档模式下
        a. 前期处理
            SQL> drop table mytest purge;
 
            SQL> drop table mytest_nolog purge;
           
            SQL> select log_mode from v$database;
 
            LOG_MODE
            ————
            ARCHIVELOG
       
        b. 创建表对象并进行比较
            SQL> create table mytest as select * from dba_objects where 1=0;    –logging 模式创建表对象
 
            SQL> create table mytest_nolog nologging as select * from dba_objects where 1=0;–nologging 模式创建表对象
 
            SQL> SELECT a.name,b.value FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic#  WHERE a.name = ‘redo size’;   
 
            NAME                VALUE
            ————— ———-    – 查看当前的 redo size 为 46844
            redo size            46844
 
            SQL> insert into mytest select * from dba_objects;    – 为表 mytest 使用常规 insert 插入记录
 
            11598 rows created.
            Elapsed: 00:00:00.25
 
            SQL> SELECT a.name,b.value FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic#  WHERE a.name = ‘redo size’;
 
            NAME                VALUE
            ————— ———-      – 表 mytest 使用常规 insert 插入记录产生的 redo size 为 1299120-46844=1252276
            redo size          1299120
 
            SQL> insert into mytest_nolog select * from dba_objects;  – 为表 mytest_nolog 使用常规 insert 插入记录
 
            11598 rows created.
            Elapsed: 00:00:00.28
 
            SQL> SELECT a.name,b.value FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic#  WHERE a.name = ‘redo size’;
 
            NAME                VALUE
            ————— ———-    – 表 mytest_nolog 使用常规 insert 插入记录产生的 redo size 为 2552880-1299120=1253760
            redo size          2552880
 
            SQL> insert /* +append */ into mytest select * from dba_objects;– 表 mytest 使用 insert append 方式
 
            11598 rows created.
            Elapsed: 00:00:00.20
 
            SQL> SELECT a.name,b.value FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic#  WHERE a.name = ‘redo size’;
 
            NAME                VALUE
            ————— ———-    – 表 mytest 使用 insert append 插入记录产生的 redo size 为 3750852-2552880=1197972
            redo size          3750852
 
            SQL> insert /* +append */ into mytest_nolog select * from dba_objects;– 表 mytest_nolog 使用 insert append��式
 
            11598 rows created.
            Elapsed: 00:00:00.18
 
            SQL> SELECT a.name,b.value FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic#  WHERE a.name = ‘redo size’;
 
            NAME                VALUE
            —————            ———-    – 表 mytest_nolog 使用 insert append 插入记录产生的 redo size 为 4948764-3750852=1197912
            redo size          4948764
       
       
   
    3. 小结
        对于表对象插入记录时,使用普通 insert 与使用 append 方式比较
            在非归档模式下,表对象在使用 nologging 模式时,两者产生的日志量相差不大,而使用 logging 模式时,常规 insert 的日志量远大于 append 方式。
            在归档模式下,表对象使用 logging 模式,两者产生的日志量相差不大。而表对象使用 nologging 模式时,则使用 append 将使得性能有所提高。
        在非归档模式下的 inesrt append 操作将是性能最高的。
        比如我们在做 dataguard、流复制、第三方工具同步时,针对数据库开启了 force logging 的方式,各种特征的表,在插入时,日志量相差是不大的
       
           
    4.direct insert append 使用时的注意事项   
        a. 当使用 insert into … values 语句时,不能够使用 append 方式
        b.append 方式为批量插入的记录,因此新插入的记录被存储在 hwm 之上,对于 hwm 之下空闲块将不会被使用。
        c. 在 append 方式插入记录后,要执行 commit,才能对表进行查询。否则会出现错误:
            ORA-12838: cannot read/modify an object after modifying it in parallel
        d. 在归档模式下,表对象具有 nologging 属性,且以 append 方式批量添加记录,才会显著减少 redo 数量。
        e. 在非归档模式下,表对象即便具有 logging 属性,也可减少 redo 数量。
        f. 对于表上具有索引的表对象,如果新增的记录数量为整个表的很少一部分,则直接以 append 方式批量添加记录,如果原表记录很少,
            实时性要求不是很高,而新增记录很多,可以先删除索引,在使用 append 方式追加记录,最后再创建索引。

更多 Oracle 相关信息见Oracle 专题页面 http://www.linuxidc.com/topicnews.aspx?tid=12

本文永久更新链接地址:http://www.linuxidc.com/Linux/2016-06/132016.htm

Oracle 数据库日志有几种模式,强制日志、非强制日志模式、对象级别等!通常在数据库模式设置强制后,余下的不管如何都会强制日志!现在我们就 oracle 数据库日志模式产生的日志量进行比较测试,在各种模式下日志产生的情况!

日志记录模式的转换
      a. 数据库从非强制日子模式切换到强制日志模式
          SQL>alter database force logging;   
        b. 数据库从强制日志模式切换到非强制日志模式
          SQL>alter database noforce logging;
        c. 表空间级别从强制日志模式切换到非强制日志模式
          SQL>alter tablespace tablespacename noforce logging;
        d. 表空间级别从非强制日志模式切换到强制日志模式
          SQL>alter tablespace tablespacename force logging;
        e. 对象级别日志记录模式
          SQL>alter tablet mytest nologging;    – 不记录日志模式
          SQL>alter tablet mytest logging;      – 采用日志记录模式
 
一、表段,索引段上使用一般 DDL,DML 时,LOGGING 与 NOLOGGING 情况 
    1. 查看数据库的归档模式
        有关设置日志归档模式的问题,请参考:
            Oracle 联机重做日志文件(ONLINE LOG FILE)
            Oracle 归档日志
 
        SQL> select log_mode,force_logging from v$database;
 
        LOG_MODE    FOR
        ———— —
        ARCHIVELOG  NO
       
        SQL> archive log list;
        Database log mode              Archive Mode
        Automatic archival            Enabled
        Archive destination            /u01/mytest/arch
        Oldest online log sequence    1024
        Next log sequence to archive  1025
        Current log sequence          1025 
       
        SQL> select tablespace_name,logging,force_logging from dba_tablespaces;
 
        TABLESPACE_NAME                LOGGING  FOR
        —————————— ——— —
        SYSTEM                        LOGGING  NO
        UNDOTBS1                      LOGGING  NO
        SYSAUX                        LOGGING  NO
        TEMP                          NOLOGGING NO
        USERS                          LOGGING  NO
        PERFSTAT                      LOGGING  NO
       

     
    2. 使用如下语句进行查询
     
          SQL >SELECT a.name,b.value  FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic# WHERE a.name = ‘redo size’;
 

 
    3. 在归档模式下比较表段上的 NOLOGGING 与 LOGGING
        SQL>  SELECT a.name,b.value  FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic# WHERE a.name = ‘redo size’;
 
        NAME                VALUE
        ————— ———-
        redo size            1644
 
        SQL> CREATE TABLE mytest_nolog NOLOGGING AS SELECT * FROM dba_objects;  –nologging 模式创建表
 
        SQL> SELECT a.name,b.value  FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic# WHERE a.name = ‘redo size’;
 
        NAME                VALUE
        ————— ———-
        redo size            70064         
 
        SQL> select 1644 last,70064 as cur,(70064-1644) diff from dual;– 使用 nologging 模式建表产生的 redo size 为
 
              LAST        CUR      DIFF
        ———- ———- ———-
              1644      70064      68420
 
        SQL> CREATE TABLE mytest LOGGING AS SELECT * FROM dba_objects;    – 使用 logging 模式来创建表
 
        SQL> SELECT a.name,b.value  FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic# WHERE a.name = ‘redo size’;;  – 查看当前的 redo size
 
        NAME                VALUE
        ————— ———-           
        redo size          1344112             

        SQL> select 70064 last,1344112 as cur,(1344112-70064) diff from dual; 
 
              LAST        CUR      DIFF
        ———- ———- ———-
            70064    1344112      1274048          —- 查看 logging 模式产生的 redo size 为 -68420=1274048, 比 nologging 日志模, 有 19 倍多!
             
 
        SQL> select table_name,logging from user_tables where table_name like ‘MYTEST%’;– 查看创建表的日志记录模式
 
        TABLE_NAME                    LOG
        —————————— —
        MYTEST                        YES
        MYTEST_NOLOG                  NO
 
    4. 基于索引来比较 redo size(同样是在归档模式下)
        SQL> SELECT a.name,b.value  FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic# WHERE a.name = ‘redo size’;;    – 查看当前的 redo_size
 
        NAME                VALUE
        ————— ———-
        redo size            1140
 
        SQL> create index idx_mytest on mytest(object_id);  – 基于表 mytest 来创建索引
 
        SQL> SELECT a.name,b.value  FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic# WHERE a.name = ‘redo size’;;      – 查看当前的 redo_size
 
        NAME                VALUE
        ————— ———-                – 基于 loggiing 模式,创建索引产生的 redo size 为 221600-1140=220460
        redo size          221600
 
        SQL> alter index idx_mytest rebuild;    – 重建索引
           
        SQL> select 221600 last,448132 cur,448132-221600 diff from dual;
       
              LAST        CUR      DIFF
        ———- ———- ———-
            221600    448132    226532  – 重建索引产生的日志比直接建还要多,主要是重建过程还有一个对旧索引的删除
 
             
        SQL> alter index idx_mytest rebuild nologging; – 使用 nologging 重建索引。
                                                – 也可以在创建索引的时候直接使用 nologging 关键字
                                                – 如:create index idx_mytest_nolog mytest_nolog(object_id) nologging
        SQL> SELECT a.name,b.value  FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic# WHERE a.name = ‘redo size’;;    – 查看当前的 redo size
 
        NAME                VALUE
        ————— ———-              – 基于 nologging 日志模式重建索引产生的 redo size 为 469160-448132=21028
        redo size          469160
 
    5. 非归档模式下的 LOGGING 与 NOLOGGING
 
        SQL> drop table mytest purge;
 
        SQL> drop table mytest_nolog purge;
       
        SQL> select log_mode,force_logging from v$database;  – 切换日志到非归档模式后,下面是查询的结果
 
        LOG_MODE    FOR
        ———— —
        NOARCHIVELOG NO
       
        SQL> SELECT a.name,b.value  FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic# WHERE a.name = ‘redo size’;;    – 查看当前的 redo size
 
        NAME                VALUE
        ————— ———-
        redo size            1688
        SQL> create table mytest as select * from dba_objects;  – 创建表对象,使用 logging 日志记录模式
 
        SQL> SELECT a.name,b.value  FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic# WHERE a.name = ‘redo size’;;  – 查看当前的 redo size
 
        NAME                VALUE
        ————— ———-            – 使用 logging 日志记录模式,创建表对象之后产生的 redo size 为 -1688 =68548
        redo size            70236
 
        SQL> create table mytest_nolog nologging as select * from dba_objects; – 创建表对象,使用 nologging 日志记录模式
 
        SQL> SELECT a.name,b.value  FROM v$statname a JOIN v$mystat b ON a.statistic# = b.statistic# WHERE a.name = ‘redo size’;;  – 查看当前的 redo size
 
        NAME                VALUE
        ————— ———-            – 使用 nologging 日志记录模式创建表对象之后产生的 redo size 为 135464-70236=65228
 
        redo size          135464
         
    6. 小结:
        使用 logging 与 nologging 来创建对象或执行 DML 时
            对于非归档模式下,其产生的日志信息(redo size) 相差的并不大
            对于归档模式下,logging 模式产生的日志将远远大于使用 nologging 模式产生的日志量

更多详情见请继续阅读下一页的精彩内容:http://www.linuxidc.com/Linux/2016-06/132016p2.htm

正文完
星哥玩云-微信公众号
post-qrcode
 0
星锅
版权声明:本站原创文章,由 星锅 于2022-01-22发表,共计16697字。
转载说明:除特殊说明外本站文章皆由CC-4.0协议发布,转载请注明出处。
【腾讯云】推广者专属福利,新客户无门槛领取总价值高达2860元代金券,每种代金券限量500张,先到先得。
阿里云-最新活动爆款每日限量供应
评论(没有评论)
验证码
【腾讯云】云服务器、云数据库、COS、CDN、短信等云产品特惠热卖中