当前位置: 首页 > 科技观察

oracle使用在线重定义给表增加新的列和分区

时间:2023-03-13 07:50:52 科技观察

因为需要修改一个上亿条记录的表我们要记录到分区表中,所以尽量使用在线重定义来增加新的列和表的分区。实现,下面是一个测试例子,操作系统为OracleLinux7.1,数据库为12.2.0.1,原始表为emp_redef,在hrscheme中存放:SQL>deschr.emp_redefNameTypeNullableDefaultComments---------------------------------------------EMPLOYEE_IDNUMBER(6)FIRST_NAMEVARCHAR2(20)YLAST_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与需要进行在线重定义操作的用户连接。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_redefSQL>createtablehr.int_emp_redef(employee(6)UMBER_idNUnotnull,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);Tablecreated4.开始重定义操作SQL>begindbms_redefinition.start_redef_table(uname=>'hr',orig_table=>'emp_table=redef''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.复制依依辞int(emp_redef创建任何触发器、索引、物化视图日志、授权和约束)SQL>declarenum_errorspls_integer;begindbms_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);结束;/PL/SQL过程成功完成。请注意,调用此过程时需要将ignore_errors参数设置为TRUE。中间表创建主键约束,执行copy_table_dependents过程试图从原表复制主键约束和索引时出错。这些错误可以忽略,但必须进行下一步的查询,看是否还有其他错误。6.查询dba_redefinition_errors视图来查看错误信息SQL>setlong8000SQL>setpages8000SQL>columnobject_nameheading'objectname'formata20SQL>columnbase_table_nameheading'basetablename'formata10SQL>columnddl_txtheading'ddlthatcausederror'formata40SQL>selectobject_name,base_table_name,ddl_txtfromdba_redefinition_errors;objectnamebasetableddlthatcausederror------------------------------------------------------------------SYS_C0023200EMP_REDEFALTERTABLE“HR”。上面的错误提示是中间表的last_name和job_id列不为null,reason表为null。这个错误可以忽略。7、同步中间表hr.int_emp_redefSQL>begindbms_redefinition.sync_interim_table(uname=>'hr',orig_table=>'emp_redef',int_table=>'int_emp_redef');结束;/PL/SQLproceduresuccessfullycompleted.8。完成重定义操作SQL>begindbms_redefinition.finish_redef_table(uname=>'hr',orig_table=>'emp_redef',int_table=>'int_emp_redef');结尾;/PL/SQL过程成功完成。表hr.emp_redef只会被短时间锁定在独占模式,以结束Redefine操作。操作完成后,将使用hr.int_emp_redef表的所有属性重新定义表hr.emp_redef。SQL>deschr.emp_redefNameTypeNullableDefaultComments-------------------------------------------------EMPLOYEE_IDNUMBER(6)FIRST_NAMEVARCHAR2(20)YLAST_NAMEVARCHAR2(25)JOB_IDVARCHAR2(10)DEPARTMENT_IDNUMBER(4)MGRNUMBER(5)YHIREDATEDATEY(sysdate)SALNUMBER(7,2)YBONUSNUMBER(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")USINGINDEXPCTFREE10INITRANS2MAXTRANS255COMPUTESTATISTICSSTORAGE(INITIAL65536NEXT1048576MINEXTENTS1MAXEXTENTS2147483645PCTINCREASE0FREELISTS1FREELISTGROUPS1BUFFER_POOLDEFAULTFLASH_CACHEDEFAULTCELL_FLASH_CACHEDEFAULT)TABLESPACE"USERS"ENABLE)PCTFREE10PCTUSED40INITRANS1MAXTRANS255STORAGE(BUFFER_POOLDEFAULTFLASH_CACHEDEFAULTCELL_FLASH_CACHEDEFAULT)TABLESPACE"USERS"PARTITIONBYRANGE("EMPLOYEE_ID")(PARTITION"EMP200"VALUESLESSTHAN(200)SEGMENTCREATIONIMMEDIATEPCTFREE10PCTUSED40INITRANS1MAXTRANS255NOCOMPRESSLOGGINGSTORAGE(INITIAL8388608NEXT1048576MINEXTENTS1MAXEXTENTS2147483645PCTINCREASE0FREELISTS1FREELISTGROUPS1BUFFER_POOLDEFAULTFLASH_CACHEDEFAULTCELL_FLASH_CACHEDEFAULT)TABLESPACE"USERS",PARTITION"EMP400"VALUESLESSTHAN(400)SEGMENTCREATIONIMMEDIATEPCTFREE10PCTUSED40INITRANS1MAXTRANS255NOCOMPRESSLLOGGINGSTORAGE(INITIAL8388608NEXT1048576MINEXTENTS1MAXEXTENTS2147483645PCTINCREASE0FREELISTS1FREELISTGROUPS1BUFFER_POOLDEFAULTFLASH_CACHEDEFAULTCELL_FLASH_CACHEDEFAULT)TABLESPACE"USERS")行已选中。可以看到hr.emp_redef表变成了在线重定义的函数,任何查询完成后都会重定义中间表。然后执行删除中间表的语句。改变原表的表结构SQL>deschr.int_emp_redefNameTypeNullableDefaultComments-----------------------------------------------EMPLOYEE_IDNUMBER(6)YFIRST_NAMEVARCHAR2(20)YLAST_NAMEVARCHAR2(25)JOB_IDVARCHAR2(10)DEPARTMENT_IDNUMBER(4)YSQL>droptablehr.int_emp_redefpurge;Tabledroppedhere,在线重定义表hr.emp_redef操作完成