1818IP-服务器技术教程,云服务器评测推荐,服务器系统排错处理,环境搭建,攻击防护等

当前位置:首页 - 数据库 - 正文

君子好学,自强不息!

因为需要将一张上亿我们要记录的表修改为分区表,所以尝试使用联机重定义来给表增加新列与分区的方法来实现,下面是一个测试的例子,操作系统是Oracle Linux 7.1,数据库为12.2.0.1,原始表为emp_redef,该表存储在hr方案中:

SQL>deschr.emp_redef
NameTypeNullableDefaultComments
------------------------------------------------
EMPLOYEE_IDNUMBER(6)
FIRST_NAMEVARCHAR2(20)Y
LAST_NAMEVARCHAR2(25)
JOB_IDVARCHAR2(10)
DEPARTMENT_IDNUMBER(4)Y

表emp_redef将按以下规则来进行联机重定义:

.增加新列mgr,hiredate,sal与bonus

.新列bonus被初始化为0

.列department_id的值由10开始增加

.表将被重定义为范围分区表,分区键为employee_id。

联机重定义操作如下:

1.用要执行联机重定义操作的用户登录数据库

SQL>connpm/pm@jypdb
Connected.

2.验证表emp_redef是否可以执行联机重定义。在这种情况下,可以使用主键或伪主键来来进行验证。

SQL>execdbms_redefinition.can_redef_table(uname=>'HR',tname=>'EMP_REDEF',options_flag=>dbms_redefinition.cons_use_pk);
PL/SQLproceduresuccessfullycompleted.

3.创建一个中间表hr.int_emp_redef

SQL>createtablehr.int_emp_redef
(
employee_idNUMBER(6)notnull,
first_nameVARCHAR2(20),
last_nameVARCHAR2(25)notnull,
job_idVARCHAR2(10)notnull,
department_idNUMBER(4)notnull,
mgrNUMBER(5),
hiredateDATEDEFAULT(sysdate),
salNUMBER(7,2),
bonusNUMBER(7,2)DEFAULT(0)
)
partitionbyrange(employee_id)
(
partitionemp200valueslessthan(200)tablespaceusers,
partitionemp400valueslessthan(400)tablespaceusers
);
Tablecreated

4.开始重定义操作

SQL>begin
dbms_redefinition.start_redef_table(
uname=>'hr',
orig_table=>'emp_redef',
int_table=>'int_emp_redef',
col_mapping=>'employee_idemployee_id,first_namefirst_name,last_namelast_name,job_idjob_id,department_id+10department_id,0bonus',
options_flag=>DBMS_REDEFINITION.CONS_USE_PK);
end;
/
PL/SQLproceduresuccessfullycompleted.

5.复制依赖对象(自动对表hr.int_emp_redef创建任何触发器,索引,物化视图日志,授权与约束)

SQL>declare
num_errorspls_integer;
begin
dbms_redefinition.copy_table_dependents(
uname=>'hr',
orig_table=>'emp_redef',
int_table=>'int_emp_redef',
copy_indexes=>DBMS_REDEFINITION.CONS_ORIG_PARAMS,
copy_triggers=>TRUE,
copy_constraints=>TRUE,
copy_privileges=>TRUE,
ignore_errors=>TRUE,
num_errors=>num_errors);
end;
/

PL/SQLproceduresuccessfullycompleted.

注意,在调用这个过程时ignore_errors参数需要设置为TRUE。原因是中间表创建了主键约束,并且当执行copye_table_dependents过程来试图从原始表复制主键约束与索引时会发生错误。可以忽略这些错误,但必须执行下一步操作中的查询来查看是否还存在其它错误。

6.查询dba_redefinition_errors视图来查看错误信息

SQL>setlong8000
SQL>setpages8000
SQL>columnobject_nameheading'objectname'formata20
SQL>columnbase_table_nameheading'basetablename'formata10
SQL>columnddl_txtheading'ddlthatcausederror'formata40
SQL>selectobject_name,base_table_name,ddl_txtfromdba_redefinition_errors;

objectnamebasetableddlthatcausederror
----------------------------------------------------------------------
SYS_C0023200EMP_REDEFALTERTABLE"HR"."INT_EMP_REDEF"MODIFY
("LAST_NAME"CONSTRAINT"TMP$$_SYS_C0023
2000"NOTNULLENABLENOVALIDATE)

SYS_C0023201EMP_REDEFALTERTABLE"HR"."INT_EMP_REDEF"MODIFY
("JOB_ID"CONSTRAINT"TMP$$_SYS_C0023201
0"NOTNULLENABLENOVALIDATE)


2rowsselected.

上面的错误信息是说中间表的last_name与job_id列为not null,而原因表为null,这种错误可以忽略。

7.同步中间表hr.int_emp_redef

SQL>begin
dbms_redefinition.sync_interim_table(
uname=>'hr',
orig_table=>'emp_redef',
int_table=>'int_emp_redef');
end;
/
PL/SQLproceduresuccessfullycompleted.

8.完成重定义操作

SQL>begin
dbms_redefinition.finish_redef_table(
uname=>'hr',
orig_table=>'emp_redef',
int_table=>'int_emp_redef');
end;
/
PL/SQLproceduresuccessfullycompleted.

表hr.emp_redef只会以排他模式被锁定很短的时间来结束重定义操作。在操作完成后,表hr.emp_redef将使用hr.int_emp_redef表的所有属性来重定义。

SQL>deschr.emp_redef
NameTypeNullableDefaultComments
--------------------------------------------------
EMPLOYEE_IDNUMBER(6)
FIRST_NAMEVARCHAR2(20)Y
LAST_NAMEVARCHAR2(25)
JOB_IDVARCHAR2(10)
DEPARTMENT_IDNUMBER(4)
MGRNUMBER(5)Y
HIREDATEDATEY(sysdate)
SALNUMBER(7,2)Y
BONUSNUMBER(7,2)Y(0)

SQL>selectdbms_metadata.get_ddl(object_type=>'TABLE',name=>'EMP_REDEF',schema=>'HR')fromdual;

DBMS_METADATA.GET_DDL(OBJECT_TYPE=>'TABLE',NAME=>'EMP_REDEF',SCHEMA=>'HR')
--------------------------------------------------------------------------------

CREATETABLE"HR"."EMP_REDEF"
("EMPLOYEE_ID"NUMBER(6,0)NOTNULLENABLE,
"FIRST_NAME"VARCHAR2(20),
"LAST_NAME"VARCHAR2(25)NOTNULLENABLE,
"JOB_ID"VARCHAR2(10)NOTNULLENABLE,
"DEPARTMENT_ID"NUMBER(4,0)NOTNULLENABLE,
"MGR"NUMBER(5,0),
"HIREDATE"DATEDEFAULT(sysdate),
"SAL"NUMBER(7,2),
"BONUS"NUMBER(7,2)DEFAULT(0),
CONSTRAINT"EMP_REDEF_EMP_ID_PK"PRIMARYKEY("EMPLOYEE_ID")
USINGINDEXPCTFREE10INITRANS2MAXTRANS255COMPUTESTATISTICS
STORAGE(INITIAL65536NEXT1048576MINEXTENTS1MAXEXTENTS2147483645
PCTINCREASE0FREELISTS1FREELISTGROUPS1
BUFFER_POOLDEFAULTFLASH_CACHEDEFAULTCELL_FLASH_CACHEDEFAULT)
TABLESPACE"USERS"ENABLE
)PCTFREE10PCTUSED40INITRANS1MAXTRANS255
STORAGE(
BUFFER_POOLDEFAULTFLASH_CACHEDEFAULTCELL_FLASH_CACHEDEFAULT)
TABLESPACE"USERS"
PARTITIONBYRANGE("EMPLOYEE_ID")
(PARTITION"EMP200"VALUESLESSTHAN(200)SEGMENTCREATIONIMMEDIATE
PCTFREE10PCTUSED40INITRANS1MAXTRANS255
NOCOMPRESSLOGGING
STORAGE(INITIAL8388608NEXT1048576MINEXTENTS1MAXEXTENTS2147483645
PCTINCREASE0FREELISTS1FREELISTGROUPS1
BUFFER_POOLDEFAULTFLASH_CACHEDEFAULTCELL_FLASH_CACHEDEFAULT)
TABLESPACE"USERS",
PARTITION"EMP400"VALUESLESSTHAN(400)SEGMENTCREATIONIMMEDIATE
PCTFREE10PCTUSED40INITRANS1MAXTRANS255
NOCOMPRESSLOGGING
STORAGE(INITIAL8388608NEXT1048576MINEXTENTS1MAXEXTENTS2147483645
PCTINCREASE0FREELISTS1FREELISTGROUPS1
BUFFER_POOLDEFAULTFLASH_CACHEDEFAULTCELL_FLASH_CACHEDEFAULT)
TABLESPACE"USERS")
rowselected.

可以看到表hr.emp_redef已经成功能联机重定义

9.等任何查询中间表的语句执行完成后将其删除,而且中间表在重定义后其结构就变成了原始表的表结构

SQL>deschr.int_emp_redef
NameTypeNullableDefaultComments
------------------------------------------------
EMPLOYEE_IDNUMBER(6)Y
FIRST_NAMEVARCHAR2(20)Y
LAST_NAMEVARCHAR2(25)
JOB_IDVARCHAR2(10)
DEPARTMENT_IDNUMBER(4)Y

SQL>droptablehr.int_emp_redefpurge;
Tabledropped

到此,联机重定义表hr.emp_redef就操作完成。

本文来源:1818IP

本文地址:https://www.1818ip.com/post/10805.html

免责声明:本文由用户上传,如有侵权请联系删除!

发表评论

必填

选填

选填

◎欢迎参与讨论,请在这里发表您的看法、交流您的观点。