oracle 11g对大表中添加DEFAULT值的NOT NULL字段速度有大幅度的提升

Oracle11g添加字段优化

今天同事问我一个问题他说在一张2000万的表上增加了一个字段并字段一个默认值,执行这条语句(alter table tablename add new_col default ‘col’)一个小时没有执行完,问我有没有其他解决方法

我查了一下资料发现

Oracle11g中,在添加一个包含DEFAULT值的NOT NULL字段,Oracle不会去更新现有的数据,Oracle需要做的不过是将默认值以及对应的表信息、列信息一起存储在一个新增数据字典表ecol$中。这张表利用BLOB字段存储ALTER TABLE添加的DEFAULT

然后我测试一下情况(在11g中测试alter table tablename add new_col default ‘col’ 后代not null和不带not null的区别)

1.新建一张表test

SQL> create table test (id number,name varchar(10));

表已创建。

2.test表中插入500000记录

SQL> begin

  2  for i in 1..500000 loop

  3  insert into test values(i,'jack');

  4  end loop;

  5  end;

  6  /

PL/SQL 过程已成功完成。

SQL> select count(1) from test;

  COUNT(1)

----------

    500000

SQL>

SQL> set timing on

3.向表中添加一个新的字段NEW_COL1带有默认值如下语句不带NOT NULL

ALTER TABLE t ADD NEW_COL1 CHAR(10) DEFAULT 'TESTCOLUMN'

SQL> ALTER TABLE test ADD NEW_COL1 CHAR(10) DEFAULT 'TESTCOLUMN';

表已更改。

已用时间00: 00: 43.23

SQL>

4. 向表中添加一个新的字段NEW_COL2带有默认值语句中带上 NOT NULL

ALTER TABLE test ADD NEW_COL2 CHAR(10) DEFAULT 'TESTCOLUMN' NOT NULL;

SQL> ALTER TABLE test ADD NEW_COL2 CHAR(10) DEFAULT 'TESTCOLUMN' NOT NULL;

表已更改。

已用时间00: 00: 00.25

惊奇的发现两者差别太大了带not null的用时不到1

这是11g对新增一个not null字段带有默认值进行了优化,默认值以及对应的表信息、列信息一起存储在一个新增数据字典表ecol$

如下:可以查询test表中添加的默认值

SQL> SELECT OBJECT_ID FROM DBA_OBJECTS WHERE OBJECT_NAME = 'TEST' and owner='IMUSE01';

OBJECT_ID

----------

74063

SQL> select COLNUM,BINARYDEFVAL from ecol$ where TABOBJ#= 74063;

COLNUM

----------

BINARYDEFVAL

--------------------------------------------------------------------------------

         4

54455354434F4C554D4E

默认值可以转换一下:

SQL> SET SERVEROUT ON

SQL> DECLARE

  2  V_BLOB VARCHAR2(32767) DEFAULT '54455354434F4C554D4E';

  3  BEGIN

  4  FOR I IN 1..LENGTH(V_BLOB)/2 LOOP

  5  DBMS_OUTPUT.PUT(CHR(TO_NUMBER(SUBSTR(V_BLOB, (I - 1) * 2 + 1, 2), 'XXX')));

  6  END LOOP;

  7  DBMS_OUTPUT.NEW_LINE;

  8  END;

  9  /

TESTCOLUMN

PL/SQL 过程已成功完成。

SQL>

Oracle数据库中,如果一张已经定义了两个主键字段,而需要新增第三个主键字段,需要注意主键约束本质上是一个唯一性约束(UNIQUE)与非空约束(NOT NULL)的组合。Oracle 不允许直接对已有主键的使用 `ALTER TABLE ... ADD PRIMARY KEY` 来新增主键字段,因为这会与现有的主键约束发生冲突。 要实现这一目标,可以按照以下步骤操作: 1. **添加新列并设置为非空** 首先向添加新的字段,并确保该字段设置为 `NOT NULL`,以满足主键的要求。 ```sql ALTER TABLE 名 ADD (新字段名 数据类型 DEFAULT 默认 NOT NULL); ``` 2. **重建主键约束** 删除原有的主键约束,然后将新字段与原有主键字段一起重新添加到主键约束中。 - 首先查询现有主键约束的名称,可以通过如下语句查询: ```sql SELECT a.constraint_name FROM user_cons_columns a, user_constraints b WHERE a.constraint_name = b.constraint_name AND b.constraint_type = 'P' AND a.table_name = '名'; ``` - 然后删除原有主键约束: ```sql ALTER TABLE 名 DROP CONSTRAINT 主键约束名称; ``` - 最后重新添加包含新字段的复合主键: ```sql ALTER TABLE 名 ADD CONSTRAINT 新主键约束名称 PRIMARY KEY (字段1, 字段2, 新字段名); ``` 通过上述方式,可以在已有的复合主键基础上添加新的主键字段[^2]。 ### 示例操作 假设有一张 `employees`,其主键为 `first_name` 和 `last_name`,现在需要添加 `phone_number` 作为第三个主键字段: ```sql -- 添加字段并设置为非空 ALTER TABLE employees ADD (phone_number VARCHAR2(20) DEFAULT '000-0000' NOT NULL); -- 查询现有主键约束名称 SELECT a.constraint_name FROM user_cons_columns a, user_constraints b WHERE a.constraint_name = b.constraint_name AND b.constraint_type = 'P' AND a.table_name = 'EMPLOYEES'; -- 删除原有主键约束(假设约束名为 EMP_PK) ALTER TABLE employees DROP CONSTRAINT EMP_PK; -- 添加新的复合主键(包含三个字段) ALTER TABLE employees ADD CONSTRAINT EMP_PK PRIMARY KEY (first_name, last_name, phone_number); ``` ### 注意事项 - 在删除主键约束之前,应确保没有其他依赖该主键的外键约束,否则会引发错误。 - 操作前建议对数据进行备份,防止误操作导致数据丢失。 - 如果中已有量数据,添加 `NOT NULL` 字段时必须指定 `DEFAULT` ,否则会因字段为空而失败。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值