浅析Oracle 11g中对数据列默认值变化的优化

日常的运维工作中,对生产数据表进行DDL操作是一件需要谨慎对待的事情。运维DBA们在进行数据DDL操作的时候,通常要全局考虑,诸如对生产影响、执行时间长度和影响存储数据等等。 数据列默认值的添加,是DBA们经常头疼的一个问题。传统的执行语句,消耗时间

在日常的运维工作中,对生产数据表进行DDL操作是一件需要谨慎对待的事情。运维DBA们在进行数据DDL操作的时候,通常要全局考虑,诸如对生产影响、执行时间长度和影响存储数据等等。

数据列默认值的添加,是DBA们经常头疼的一个问题。传统的执行语句,消耗时间长、资源使用量大,对生产环境影响程度高。采用其他的一些变通方法,又存在操作步骤繁琐的问题。如何快速的添加一个有默认值的数据列,同时对现有生产环境影响最小,是我们希望达到的一个目标。

本文从操作入手,探讨添加default数据列的问题点,最后介绍Oracle 11g中对其进行的“革命性”优化

1、从10g的数据列添加谈起

为了实现对比效果,我们首选选择10g版本的Oracle进行试验,构造一个相对较大的数据表。

SQL> select * from v$version;
BANNER
—————————————————————-
Oracle Database 10g Enterprise Edition Release 10.2.0.1.0 – Prod
PL/SQL Release 10.2.0.1.0 – Production
CORE   10.2.0.1.0     Production
TNS for 32-bit Windows: Version 10.2.0.1.0 – Production
NLSRTL Version 10.2.0.1.0 – Production

SQL> create table t as select object_id from dba_objects;
表已创建。

SQL> select count(*) from t;
COUNT(*)
———-
  3220352

数据表t只包括一个数据列,但是数据量大约为320万条。我们从体积上进行评估如下:

SQL> set timing on;
SQL> select bytes/1024/1024,blocks from dba_segments where wner=’SYS’ and segment_name=’T’;

BYTES/1024/1024    BLOCKS
————— ———-
            39      4992

已用时间: 00: 00: 00.03

SQL> exec dbms_stats.gather_table_stats(user,’T’,cascade=>true);
PL/SQL过程已成功完成。

已用时间: 00: 00: 00.35

SQL> select blocks from dba_tables where wner=’SYS’ and table_name=’T’;

   BLOCKS
———-
     4883

已用时间: 00: 00: 00.01

Oracle分配给这个段segment的中空间为4992个数据块,高水位线HWM下的格式化过数据块为4883。总体积约40M。

下面进行两种方式的添加数据表默认值列方法,一起观察一下变化情况。首先是允许为空默认值列的操作。

SQL> alter table t add vc varchar2(100) default ‘TTTTTTTTTTTT’;

表已更改。

已用时间: 00: 34: 37.15

SQL> exec dbms_stats.gather_table_stats(user,’T’,cascade=>true);

PL/SQL过程已成功完成。

已用时间: 00: 00: 03.86

SQL> select bytes/1024/1024,blocks from dba_segments where wner=’SYS’ and segment_name=’T’;

BYTES/1024/1024    BLOCKS
————— ———-
           208     26624

已用时间: 00: 00: 00.06
SQL> select blocks from dba_tables where wner=’SYS’ and table_name=’T’;

   BLOCKS
———-
    25864

已用时间: 00: 00: 00.01

果然是一个费时的操作,添加一个数据列默认值,总共消耗了近30分钟时间。原有数据表的体积也发生的膨胀,从原来的不到40M,上升到了208M。

这个现象告诉我们,当我们添加一个有default值的数据列,并且是直接添加的时候,一些数据被插入到了数据块中,引起空间膨胀。

在原有的结构下,数据添加到数据块上是必需的,只有这样才能将数据列default添加到里面去。

除了这个字句,我们是还可以提供数据列的not null选项,也是可以实现相同的功能的。

SQL> alter table t add vc2 varchar2(100) default ‘TTTTTTTTTTTT’ not null;

表已更改。

已用时间: 00: 15: 58.85

SQL> exec dbms_stats.gather_table_stats(user,’T’,cascade=>true);

PL/SQL过程已成功完成。

已用时间: 00: 00: 36.87

SQL> select bytes/1024/1024,blocks from dba_segments where wner=’SYS’ and segme
nt_name=’T’;

BYTES/1024/1024    BLOCKS
————— ———-
           256     32768

已用时间: 00: 00: 00.14
SQL> select blocks from dba_tables where wner=’SYS’ and table_name=’T’;

   BLOCKS
———-
    32448

已用时间: 00: 00: 00.04

也是消耗了15分钟,空间发生了很大程度变化。新空间分配,同时数据行数没有发生变化,潜在的行迁移(Row Migration)和行链接(Row Chaining)是严重恶化的!

综合分析Oracle 10g下的操作:为了添加上数据字段的默认值,Oracle会去访问每个数据块上的每个数据行进行数据列拓展工作,这个过程中还伴随着新空间分配和多余数据行复制。

这类型操作对于生产环境是恐怖的,在整个作业过程中,数据表结构被锁定,相关业务处理操作阻塞或者缓慢。所以,运维DBA都是选择在维护窗口或者变通的方法进行处理。

在Oracle 11g环境下,事情有了一些不同。

2、11g下的默认值配置

我们在11g上进行相似操作。

SQL> select * from v$version;
BANNER
——————————————————————————–
Oracle Database 11g Enterprise Edition Release 11.2.0.1.0 – Production

PL/SQL Release 11.2.0.1.0 – Production
CORE       11.2.0.1.0        Production

构建相似规模的数据表。

SQL> set timing on;
SQL> create table t as select object_id from dba_objects;
Table created

SQL> select count(*) from t;
COUNT(*)
———-
  3323167

SQL> exec dbms_stats.gather_table_stats(user,’T’,cascade => true);
PL/SQL procedure successfully completed

SQL> select segment_name, bytes/1024/1024, extents,blocks from user_segments where segment_name=’T’;

SEGMENT_NA BYTES/1024/1024   EXTENTS    BLOCKS
———- ————— ———- ———-
T                      40        55      5120

SQL> select NUM_ROWS, BLOCKS from dba_tables where wner=’SCOTT’ and table_name=’T’;

NUM_ROWS    BLOCKS
———- ———-
  3323167      5041

11g下我们准备了约330万数据,进行添加非空带默认值的数据列。

SQL> alter table t add vc2 varchar2(100) default ‘TTTTTTTTTTTT’ ;

alter table t add vc2 varchar2(100) default ‘TTTTTTTTTTTT’

ORA-01013:用户请求取消当前的操作

在添加defalut列,不指定not null的时候,数据持续时间超过了我们的想象。笔者主动将其断开了。下面试试添加not null时候。

–1s不到完成操作;
SQL> alter table t add vc varchar2(100) default ‘TTTTTTTTTTTT’ not null;
Table altered

Executed in 0.047 seconds

SQL> exec dbms_stats.gather_table_stats(user,’T’,cascade => true);
PL/SQL procedure successfully completed

SQL> select NUM_ROWS, BLOCKS from dba_tables where wner=’SCOTT’ and table_name=’T’;
NUM_ROWS    BLOCKS
———- ———-
  3323167      5041

Executed in 0 seconds

SQL> select segment_name, bytes/1024/1024, extents,blocks from user_segments where segment_name=’T’;

SEGMENT_NA BYTES/1024/1024   EXTENTS    BLOCKS
———- ————— ———- ———-
T                      40        55      5120

SQL> select * from t where rownum

OBJECT_ID VC
———- ——————————————————————————–
       20 TTTTTTTTTTTT
       46 TTTTTTTTTTTT
       28 TTTTTTTTTTTT
       15 TTTTTTTTTTTT
(篇幅原因,有省略……)
9 rows selected

我们发现,当执行not null的时候,Oracle以超乎想象的速度完成了过程。并且注意:数据表的体积没有发生任何变化!!但是,我们检查数据表的时候,却发现了对应列的默认值已经添加。

这个事情是比较奇怪的,有一个道理必然是可以说通:就是这个默认值在执行过程中,是绝对没有真正添加到数据块中的,因为只有这样才不会影响数据段的体积。

3、11g默认值处理的优化

那么,11g这个过程中是如何处理的呢?而且为什么只有添加Not null的时候才会有这个特点。我们从select数据行的trace进行入手。

我们选择10046跟踪一下select的全过程,看看显示出来的默认值从哪里来。

SQL> select value from v$diag_info where name=’Default Trace File’;

VALUE
———————————————————————–
/u01/diag/rdbms/wilson/wilson/trace/wilson_ora_6177.trc

SQL> alter session set events ‘10046 trace name context forever, level 12’;
会话已更改。

SQL> select * from t where rownum

OBJECT_ID
———-
VC
—————————————————————————–

SQL> alter session set events ‘10046 trace name context off’;
会话已更改。

对生成的trace文件进行处理,获取到tkprof结果。

D:\des>tkprof wilson_ora_6177.trc
output = res.txt

TKPROF: Release 10.2.0.1.0 – Production on星期五8月24 22:07:10 2012
Copyright (c) 1982, 2005, Oracle. All rights reserved.

在分析的结果中,我们发现很多的recursive语句,也就是Oracle为了执行这个SQL,连带运行了很多的语句,其中我们发现了一个“可疑”对象。

***********************************************************************

select binaryDefVal, length(binaryDefVal)
from
ecol$          where tabobj# = :1 and colnum = :2

call    count      cpu   elapsed      disk     query   current       rows
——- —— ——– ———- ———- ———- ———- ———-
Parse       1     0.00      0.00         0         0         0          0
Execute     1     0.00      0.00         0         0         0          0
Fetch       1     0.00      0.00         2         2         0          1
——- —— ——– ———- ———- ———- ———- ———-
total       3     0.00      0.00         2         2         0          1

ecol$是sys用户下的一个新添加的字典基表,其中内容如下:

SQL> desc ecol$;
Name        Type  Nullable Default Comments
———— —— ——– ——- ——–
TABOBJ#     NUMBER Y                       
COLNUM      NUMBER Y                       
BINARYDEFVAL BLOB  Y                       

SQL> select * from ecol$;

  TABOBJ#    COLNUM BINARYDEFVAL
———- ———- ————
    76046         2
Executed in 0.031 seconds

SQL> col owner for a10;
SQL> col object_name for a10;
SQL> select owner, object_name, object_id from dba_objects where object_id in (76046);

OWNER     OBJECT_NAM OBJECT_ID
———- ———- ———-
SCOTT     T              76046

Executed in 0 seconds

从ecol$数据表中,我们发现了对数据表T对象第二列(column=2)的一个对象引用,引用的值binarydefval是一个blob类型。从直观上,我们已经可以猜出这个就是记录了数据表vc列的默认值。

此处,我们说一个问题,在Oracle中,默认值都是通过大对象类型进行保存。在数据字典col$中,默认值是通过long类进行保存。而进入11g的ecol$表,这个值是使用blob类型进行保存。

另一个需要注意的,就是这个数据表中只有一个数据行,也就是只有我们创建数据表T的默认值。这说明什么呢?

此时,我们已经可以猜出Oracle的良苦用心。首先,Oracle注意到了在生产online的时候,添加带默认值列数据的困难。但是,从现有的体系结构和存储结构下,将默认值逐行插入、从而引起行迁移的情况是不能避免的。所以,Oracle采用了一种“障眼法”。

如果我们在创建数据表的时候就指定了数据列的默认值、或者没有要求将所有数据空值一次性全都变成默认值的时候,Oracle还是按照原有的存储策略进行管理。如果出现了要求添加数据列,并且一次性将所有默认值列都加入的情况,Oracle索性就不进行插入数据和挪行的操作,而是将这个默认值保存在ecol$中。

接下来,如果要进行检索数据,首先Oracle会利用recursive call的方法,保存提取出默认值。在检索数据的过程中,如果遇到默认值列为空的情况(没有插值),就将取出的默认值输出到界面上进行显示。其实,数据行对应的默认值列是没有这个值的。

这就解释了为什么只有在添加not null默认值列的时候,才会有这个优化。因为Oracle需要确认这个列不会有空值,才会将出现的空值全都进行“障眼法”匹配。

4、结论

借助了11g这个特性,我们说在online生产环境下,临时加入默认值列就不是一件恐怖的工作了。不过,处于谨慎的考虑,还是希望有条件的时候,将该数据表进行重构。这种特性属于应急环境下考虑使用。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/467102.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月8日 05:52:10
下一篇 2025年11月8日 05:54:54

相关推荐

  • 优化CSS解析过程中的回流和重绘技巧

    CSS回流和重绘解析及优化技巧 近年来,网页性能优化成为了前端开发中的重要环节,其中包括对CSS回流和重绘的解析及优化。在优化CSS的过程中,我们需要了解回流和重绘的定义,并学习一些具体的优化技巧。 什么是回流和重绘? 回流(reflow)和重绘(repaint)是浏览器渲染引擎对网页进行布局和绘制…

    2025年12月24日
    000
  • 优化网页加载速度的技巧:理解回流和重绘的差异与优化方法

    回流与重绘的差异与优化:优化网页加载速度的技巧 在如今互联网高速发展的时代,网页加载速度成了用户体验的重要指标之一。加载速度慢不仅会让用户感到不耐烦,还会导致用户流失,影响网站的转化率。而要提高网页的加载速度,我们就需要了解和优化回流与重绘。 回流(reflow)和重绘(repaint)是浏览器渲染…

    2025年12月24日
    300
  • 提高页面渲染速度:优化回流和重绘的关键方法

    提高页面渲染速度:优化回流和重绘的关键方法,需要具体代码示例 随着网页应用的发展,用户对页面加载速度的要求也越来越高。而页面的渲染速度受到回流和重绘的影响,因此我们需要优化这两个过程来提高页面的渲染速度。本文将介绍一些关键的方法,并提供具体的代码示例。 使用transform替代top/left当改…

    2025年12月24日
    000
  • 通过使用Web标准,提升网页性能与用户体验的方法

    随着互联网的快速发展,越来越多的企业和个人都开始关注网页的性能和用户体验。一方面,良好的网页性能可以提高网站的可访问性和搜索引擎排名,另一方面,优秀的用户体验可以增加用户的黏性和转化率。而借助Web标准来优化网页性能与用户体验,则成为现如今的一种主流方法。 那么,如何利用Web标准来优化网页性能与用…

    2025年12月24日
    000
  • 比较重排、重绘和回流的优化策略以提高网页性能

    优化网页性能:探讨重排、重绘和回流的优劣比较,需要具体代码示例 随着互联网的发展,网页性能优化已成为每个前端开发人员需要面对的一个重要问题。在优化网页性能的过程中,我们需要了解并针对不同的操作进行优化。其中,重排、重绘和回流是导致网页性能下降的常见问题,本文将探讨它们的优劣,并给出一些具体的代码示例…

    2025年12月24日
    000
  • 使用关系型选择器优化CSS选择器:提升选择效率的技巧

    优化CSS选择器:如何使用关系型选择器提高选择效率 引言:在前端开发中,CSS选择器是一个非常重要的概念。它用来为HTML元素添加样式,控制页面的外观和布局。然而,在大型项目中,优化CSS选择器的效率显得尤为重要。本文将介绍如何使用关系型选择器来提高选择效率,并附上具体的代码示例。 一、什么是关系型…

    2025年12月24日
    000
  • 优化网页排版的CSS属性使用指南

    优化网页排版的CSS属性使用指南 在现代网页设计中,好的排版是不可或缺的一部分。正确使用CSS属性可以有效地改善网页排版的质量和用户体验。本文将为您介绍一些常用的CSS属性以及示例代码,帮助您优化网页排版。 一、字体属性 font-size:控制字体的大小,可以使用像素、百分比或者em作为单位。例如…

    2025年12月24日
    000
  • CSS 清除样式属性优化技巧:reset 和 normalize

    CSS 清除样式属性优化技巧:reset 和 normalize 在开发网页时,经常会遇到浏览器默认样式的干扰,导致网页显示效果不一致。为了解决这个问题,我们可以使用 CSS 清除样式属性的优化技巧。本文将介绍两种常用的方式:reset 和 normalize,并提供具体的代码示例。 一、Reset…

    2025年12月24日
    000
  • 优化用户界面体验的秘密武器:CSS开发项目经验大揭秘

    在当今数字化的时代,网站和应用程序的用户界面体验对于吸引和留住用户至关重要。而在开发用户界面时,CSS是一种不可或缺的技术。CSS(层叠样式表)是一种用来描述网页样式的语言,通过CSS,我们可以控制网页的布局、字体、颜色、动画等方方面面。然而,要想真正实现一个优秀的用户界面体验,只掌握基本的CSS语…

    2025年12月24日
    000
  • CSS 响应式图像属性优化技巧:max-width 和 object-fit

    CSS 响应式图像属性优化技巧:max-width 和 object-fit 在设计响应式网页时,优化图像是至关重要的一环。图像的处理不仅影响页面的加载速度,还会影响用户体验。在传统的网页开发中,经常会使用 max-width 属性来实现图像的响应式调整,但这往往会导致图像变形或者失真。而近年来引入…

    2025年12月24日
    000
  • CSS 径向渐变属性优化技巧:radial-gradient 和 background-position

    CSS 径向渐变属性优化技巧:radial-gradient 和 background-position 引言:CSS 径向渐变(radial-gradient)是一种用于创建圆形渐变效果的属性,常用于设计网页的背景、按钮样式等。在使用径向渐变时,结合合理的 background-position …

    2025年12月24日
    000
  • CSS 动画属性优化技巧:animation 和 transition

    CSS 动画属性优化技巧:animation 和 transition 引言:随着 Web 技术的不断发展,CSS 动画成为了网页设计和开发中非常重要的一部分。在过去,开发者通常使用 JavaScript 来实现动画效果,但现在通过 CSS 动画属性,我们可以更加轻松和高效地创建各种动画效果。本文将…

    2025年12月24日
    000
  • CSS 形状属性优化技巧:border-radius 和 clip-path

    CSS 形状属性优化技巧:border-radius 和 clip-path 在CSS中,我们经常使用一些属性来调整元素的形状,以使其更加吸引人和视觉上的吸引力。其中两个常用的属性是border-radius和clip-path。本文将详细介绍这两个属性,并提供一些优化技巧,以及具体的代码示例。 一…

    2025年12月24日
    000
  • CSS 布局属性优化技巧:position sticky 和 flexbox

    CSS 布局属性优化技巧:position sticky 和 flexbox 在网页开发中,布局是一个非常重要的方面。良好的布局结构可以提高用户体验,使页面更加美观和易于导航。而CSS布局属性则是实现这一目标的关键。在本文中,我将介绍两种常用的CSS布局属性优化技巧:position sticky和…

    2025年12月24日
    000
  • CSS 清除浮动属性优化技巧:clear 和 overflow

    CSS 清除浮动属性优化技巧:clear 和 overflow 在前端开发中,常常会遇到浮动元素造成布局混乱的情况。浮动元素可以实现元素在页面中左浮、右浮或居中浮动的效果,但它也可能导致父元素高度塌陷、布局错乱等问题。为了解决这些问题,我们需要使用一些技巧来清除浮动属性。本文将介绍两种常用的清除浮动…

    2025年12月24日
    100
  • 如何使用Css Flex 弹性布局优化移动端网页加载速度

    如何使用CSS Flex弹性布局优化移动端网页加载速度 随着移动设备的普及和互联网的快速发展,移动端网页加载速度成为了开发人员需要重视的问题之一。网页加载速度的快慢直接影响用户体验和网站的流量。在移动端网页的布局方面,CSS Flex弹性布局是一个值得开发人员注意的技术,它可以帮助我们更好地优化移动…

    2025年12月24日
    000
  • 如何优化CSS Positions布局以提升搜索引擎友好性

    如何优化CSS Positions布局以提升搜索引擎友好性 在网站开发过程中,搜索引擎优化(SEO)是至关重要的一环。除了关键词的优化和网站内容的质量之外,布局的优化也是提升搜索引擎友好性的重要因素之一。而CSS的布局选择则对网站的搜索引擎友好性有着直接的影响。本文将介绍如何优化CSS Positi…

    2025年12月24日
    000
  • 运用CSS3样式优化网页加载速度的实用方法

    运用CSS3样式优化网页加载速度的实用方法 随着互联网的快速发展,网页加载速度成为用户体验的重要指标之一。在许多情况下,用户会因为网页加载缓慢而选择离开。为了解决这个问题,前端开发人员可以通过优化CSS3样式来提高网页的加载速度。本文将介绍一些实用的方法,帮助开发人员在保持设计美观的同时,改善网页的…

    2025年12月24日
    000
  • CSS3编程优化技巧:如何巧用is与where选择器

    CSS 是前端开发人员必备的技能之一,而 CSS3 则是 CSS 的进阶版本,包含了更多强大的特性和功能。在 CSS3 中,is 选择器和 where 选择器是两个非常实用的编程优化技巧,能够帮助开发人员更高效地编写样式代码。 一、介绍is选择器 is选择器是 CSS3 中新增的一个选择器,它能够同…

    2025年12月24日
    000
  • is与where选择器:优化CSS代码充满劲头

    is与where选择器:优化CSS代码充满劲头 在Web开发中,优化CSS代码是确保网站性能和用户体验的一个关键方面。一个优化的CSS代码可以使页面加载更快、渲染更顺畅,同时还能减少代码的复杂性和维护成本。本文将介绍is和where选择器,这两种选择器可以帮助我们优化CSS代码,让我们的代码更加精简…

    2025年12月24日
    000

发表回复

登录后才能评论
关注微信