有一系列普通表都有几十到几百GB这么大,数据从几亿到几十亿,现在想将这些表改造成分区表,用其中的时间或者其他字段来做分区,允许有一段停机时间来停这些表相关的应用,该如何做呢?
思路:
新建一张分区表,按日期建分区,确保分区表各字段和属性都和普通表一样。然后停应用,将普通表记录插入到分区表中。然后将普通表重命名,分区表命名成原表的名字,完成任务。
将原表重命名为_yyyymmdd格式的表名:
create or replace procedure p_rename_001(p_tab in varchar2) as /* 功能:将原表重命名为_yyyymmdd格式的表名 完善点:要考虑RENMAE的目标表已存在的情况,先做判断 */ v_cnt_re_tab number(9) := 0; v_sql_p_rename varchar2(4000); yyyymmdd varchar2(8); begin select to_char(sysdate, 'yyyymmdd') into yyyymmdd from dual; select count(*) into v_cnt_re_tab from user_objects where object_name = upper(p_tab || '_' || yyyymmdd); if v_cnt_re_tab = 0 then v_sql_p_rename := 'rename ' || p_tab || ' to ' || p_tab || '_' || yyyymmdd; -- DBMS_OUTPUT.PUT_LINE(v_sql_p_rename);--调试使用 p_insert_log(p_tab, 'P_RENAME', v_sql_p_rename, '完成原表的重命名,改为_YYYYMMDD形式', 1); execute immediate (v_sql_p_rename); --这里无须做判断,rename动作真实完成!如果后续只是为生成脚本而不是真实执行分区操作,最后再把这个表RENAME回去! else raise_application_error(-20066, '备份表' || p_tab || '_' || yyyymmdd || '已存在,请先删除或重命名该备份表后再继续执行!'); -- DBMS_OUTPUT.PUT_LINE('备份表'||P_TAB||'_'||YYYYMMDD||'已存在'); end if; dbms_output.put_line('操作步骤1(备份原表)-------将' || p_tab || ' 表RENMAE成 ' || p_tab || '_' || yyyymmdd || ',并删除其约束索引等'); end p_rename_001;
用CREATE TABLE AS SELECT 的方式从RENAME的_yyyymmdd表中新建出一个只有MAXVALUE的初步分区表:
create or replace procedure p_ctas_002 ( p_tab in varchar2, p_struct_only in number, p_deal_flag in number, p_part_colum in varchar2, p_parallel in number default 4, p_tablespace in varchar2 ) as /* 功能:用CREATE TABLE AS SELECT 的方式从RENAME的_yyyymmdd表中新建出一个只有MAXVALUE的初步分区表 完善点:要考虑并行,nologging 的提速方式,也要考虑最终将NOLOGGING和PARALLEL恢复成正常状态 */ v_sql_p_ctas varchar2(4000); begin v_sql_p_ctas := 'create table ' || p_tab || ' partition by range ( ' || p_part_colum || ' ) (' || ' partition P_MAX values less than (maxvalue))' || ' nologging parallel 4 tablespace ' || p_tablespace || ' as select /*+parallel(t,' || p_parallel || ')*/ *' || ' from ' || p_tab || '_' || yyyymmdd; if p_struct_only = 0 then v_sql_p_ctas := v_sql_p_ctas || ' where 1=2'; else v_sql_p_ctas := v_sql_p_ctas || ' where 1=1'; end if; --DBMS_OUTPUT.PUT_LINE(v_sql_p_ctas);--调试使用 p_insert_log(p_tab, 'p_ctas', v_sql_p_ctas, '完成CTAS建初步分区表', 2, 1); p_if_judge(v_sql_p_ctas, p_deal_flag); v_sql_p_ctas := 'alter table ' || p_tab || ' logging'; p_insert_log(p_tab, 'p_ctas', v_sql_p_ctas, '将新分区表修改回LOGGING属性', 2, 2); p_if_judge(v_sql_p_ctas, p_deal_flag); v_sql_p_ctas := 'alter table ' || p_tab || ' noparallel'; p_insert_log(p_tab, 'p_ctas', v_sql_p_ctas, '将新分区表修改回NOPARALLEL属性', 2, 3); p_if_judge(v_sql_p_ctas, p_deal_flag); dbms_output.put_line('操作步骤2(建分区表)-------通过CTAS的方式从 ' || p_tab || '_' || yyyymmdd || ' 中新建' || p_tab || '表,完成初步分区改造工作'); end p_ctas_002;
对分区表进行分区SPLIT工作
create or replace procedure p_split_part_003 ( p_tab in varchar2, p_deal_flag in number, p_part_nums in number default 24, p_tab_tablespace in varchar2 ) as /* 功能:用CREATE TABLE AS SELECT 的方式新建出一个只有MAXVALUE的初步分区表进行SPLIT, 按月份进行切分,默认p_part_nums产生24个分区,构造2年的分区表,第一个分区为当前月的 上一个月 */ v_first_day date; v_next_day date; v_prev_day date; v_sql_p_split_part varchar2(4000); begin select to_date(to_char(sysdate, 'yyyymm') || '01', 'yyyymmdd') into v_first_day from dual; for i in 1 .. p_part_nums loop select add_months(v_first_day, i) into v_next_day from dual; select add_months(v_next_day, -1) into v_prev_day from dual; v_sql_p_split_part := 'alter table ' || p_tab || ' split partition p_MAX at ' || '(to_date(''' || to_char(v_next_day, 'yyyymmdd') || ''',''yyyymmdd''))' || 'into (partition PART_' || to_char(v_prev_day, 'yyyymm') || ' tablespace ' || p_tab_tablespace || ', partition p_MAX)'; -- DBMS_OUTPUT.PUT_LINE(v_sql_p_split_part);--调试使用 p_insert_log(p_tab, 'p_split_part', v_sql_p_split_part, '分区表完成分区SPLIT工作', 3, i); p_if_judge(v_sql_p_split_part, p_deal_flag); end loop; dbms_output.put_line('操作步骤3(分区操作)-------对新建的' || p_tab || '分区表完成分区SPLIT工作'); end p_split_part_003;
从_YYYYMMDD备份表中得到表的注释,为新分区表的表名增加注释
create or replace procedure p_tab_comments_004 ( p_tab in varchar2, p_deal_flag in number ) as /* 功能:从_YYYYMMDD备份表中得到表的注释,为新分区表的表名增加注释 */ v_sql_p_tab_comments varchar2(4000); v_cnt number; begin select count(*) into v_cnt from user_tab_comments where table_name = upper(p_tab) || '_' || yyyymmdd and comments is not null; if v_cnt > 0 then for i in (select * from user_tab_comments where table_name = upper(p_tab) || '_' || yyyymmdd and comments is not null) loop v_sql_p_tab_comments := 'comment on table ' || p_tab || ' is ' || '''' || i.comments || ''''; -- DBMS_OUTPUT.PUT_LINE(v_sql_p_deal_tab_comments);--调试使用 p_insert_log(p_tab, 'p_deal_comments', v_sql_p_tab_comments, '将新分区表的表的注释加上', 4, 1); p_if_judge(v_sql_p_tab_comments, p_deal_flag); end loop; dbms_output.put_line('操作步骤4(表的注释)-------对' || p_tab || '表增加表名的注释内容'); else dbms_output.put_line('操作步骤4(表的注释)-------' || upper(p_tab) || '_' || yyyymmdd || '并没有表注释!'); end if; end p_tab_comments_004;
从_YYYYMMDD备份表中得到表和字段的注释,为新分区表的表名和字段增加注释
create or replace procedure p_col_comments_005 ( p_tab in varchar2, p_deal_flag in number ) as /* 功能:从_YYYYMMDD备份表中得到表和字段的注释,为新分区表的表名和字段增加注释 */ v_sql_p_col_comments varchar2(4000); v_cnt number; begin select count(*) into v_cnt from user_col_comments where table_name = upper(p_tab) || '_' || yyyymmdd and comments is not null; if v_cnt > 0 then for i in (select * from user_col_comments where table_name = upper(p_tab) || '_' || yyyymmdd and comments is not null) loop v_sql_p_col_comments := 'comment on column ' || p_tab || '.' || i.column_name || ' is ' || '''' || i.comments || ''''; p_insert_log(p_tab, 'p_deal_col_comments', v_sql_p_col_comments, '将新分区表的列的注释加上', 5, 1); p_if_judge(v_sql_p_col_comments, p_deal_flag); end loop; dbms_output.put_line('操作步骤5(列的注释)-------对' || p_tab || '表增加列名及字段的注释内容'); else dbms_output.put_line('操作步骤5(列的注释)-------' || upper(p_tab) || '_' || yyyymmdd || '并没有列注释!'); end if; end p_col_comments_005;
从_YYYYMMDD备份表中得到原表的DEFAULT值,为新分区表的表名和字段增加DEFAULT值
create or replace procedure p_defau_and_null_006 ( p_tab in varchar2, p_deal_flag in number ) as /* 功能:从_YYYYMMDD备份表中得到原表的DEFAULT值,为新分区表的表名和字段增加DEFAULT值 */ v_sql_defau_and_null varchar2(4000); v_cnt number; begin select count(*) into v_cnt from user_tab_columns where table_name = upper(p_tab) || '_' || yyyymmdd and data_default is not null; if v_cnt > 0 then for i in (select * from user_tab_columns where table_name = upper(p_tab) || '_' || yyyymmdd and data_default is not null) loop v_sql_defau_and_null := 'alter table ' || p_tab || ' modify ' || i.column_name || ' default ' || i.data_default; p_insert_log(p_tab, 'p_deal_default', v_sql_defau_and_null, '将新分区表的默认值加上', 6); p_if_judge(v_sql_defau_and_null, p_deal_flag); end loop; dbms_output.put_line('操作步骤6(空和默认)-------对' || p_tab || '表完成默认DEFAULT值的增加'); else dbms_output.put_line('操作步骤6(空和默认)-------' || upper(p_tab) || '_' || yyyymmdd || '并没有DEFAULT或NULL值!'); end if; end p_defau_and_null_006;
从_YYYYMMDD备份表中得到原表的CHECK值,为新分区表增加CHECK值
create or replace procedure p_check_007 ( p_tab in varchar2, p_deal_flag in number ) as /* 功能:从_YYYYMMDD备份表中得到原表的CHECK值,为新分区表增加CHECK值 另注: user_constraints已经进行了非空的判断,可以略去如下类似的从user_tab_columns获取非空判断的代码编写来判断是否 for i in (select * from user_tab_columns where table_name=UPPER(P_TAB)||'_' ||YYYYMMDD and nullable='N') loop v_sql:='alter table '||p_tab||' modify '||i.COLUMN_NAME ||' not null'; */ v_sql_p_check varchar2(4000); v_cnt number; begin select count(*) into v_cnt from user_constraints where table_name = upper(p_tab) || '_' || yyyymmdd and constraint_type = 'C'; if v_cnt > 0 then for i in (select * from user_constraints where table_name = upper(p_tab) || '_' || yyyymmdd and constraint_type = 'C') loop v_sql_p_check := 'alter table ' || p_tab || '_' || yyyymmdd || ' drop constraint ' || i.constraint_name; p_insert_log(p_tab, 'p_deal_check', v_sql_p_check, '将备份出来的原表的CHECK删除', 7, 1); p_if_judge(v_sql_p_check, p_deal_flag); v_sql_p_check := 'alter table ' || p_tab || ' ADD CONSTRAINT ' || i.constraint_name || ' CHECK (' || i.search_condition || ')'; p_insert_log(p_tab, 'p_deal_check', v_sql_p_check, '将新分区表的CHECK加上', 7, 2); p_if_judge(v_sql_p_check, p_deal_flag); end loop; dbms_output.put_line('操作步骤7(check约束)-------对' || p_tab || '完成CHECK的约束'); else dbms_output.put_line('操作步骤7(check约束)-----' || upper(p_tab) || '_' || yyyymmdd || '并没有CHECK!'); end if; end p_check_007;
从_YYYYMMDD备份表中得到原表的索引信息,为新分区表增加普通索引(唯一和非唯一索引,函数索引暂不考虑),并删除旧表索引
create or replace procedure p_index_008 ( p_tab in varchar2, p_deal_flag in number, p_idx_tablespace in varchar2 ) as /* 功能:从_YYYYMMDD备份表中得到原表的索引信息,为新分区表增加普通索引(唯一和非唯一索引,函数索引暂不考虑),并删除旧表索引 难点:需要考虑联合索引的情况 */ v_sql_p_normal_idx varchar2(4000); v_cnt number; begin select count(*) into v_cnt from user_indexes where table_name = upper(p_tab) || '_' || yyyymmdd and index_type = 'NORMAL' and index_name not in (select constraint_name from user_constraints); if v_cnt > 0 then for i in (with t as (select c.*, i.uniqueness from user_ind_columns c, (select distinct index_name, uniqueness from user_indexes where table_name = upper(p_tab) || '_' || yyyymmdd and index_type = 'NORMAL' and index_name not in (select constraint_name from user_constraints)) i where c.index_name = i.index_name) select index_name, table_name, uniqueness, max(substr(sys_connect_by_path(column_ name, ','), 2)) str ---考虑组合索引的情况 from (select column_name, index_name, table_name, row_number() over(partition by index_name, table_name order by column_name) rn, uniqueness from t) t start with rn = 1 connect by rn = prior rn + 1 and index_name = prior index_name group by index_name, t.table_name, uniqueness) loop v_sql_p_normal_idx := 'drop index ' || i.index_name; p_insert_log(p_tab, 'p_deal_normal_idx', v_sql_p_normal_idx, '删除原表索引', 8, 1); p_if_judge(v_sql_p_normal_idx, p_deal_flag); dbms_output.put_line('操作步骤8(处理索引)-------将' || i.table_name || '的' || i.str || '列的索引' || i.index_name || '删除完毕'); if i.uniqueness = 'UNIQUE' then v_sql_p_normal_idx := 'CREATE UNIQUE INDEX ' || i.index_name || ' ON ' || p_tab || '(' || i.str || ')' || ' tablespace ' || p_idx_tablespace; elsif i.uniqueness = 'NONUNIQUE' then v_sql_p_normal_idx := 'CREATE INDEX ' || i.index_name || ' ON ' || p_tab || ' (' || i.str || ')' || ' LOCAL tablespace ' || p_idx_tablespace; end if; p_insert_log(p_tab, 'p_deal_normal_idx', v_sql_p_normal_idx, '将新分区表的索引加上', 8, 2); p_if_judge(v_sql_p_normal_idx, p_deal_flag); dbms_output.put_line('操作步骤8(处理索引)-------对' || p_tab || '新分区表' || i.str || '列增加索引' || i.index_name); end loop; else dbms_output.put_line('操作步骤8(处理索引)-------' || upper(p_tab) || '_' || yyyymmdd || '并没有索引(索引模块并不含主键判断)!'); end if; end p_index_008;
从_YYYYMMDD备份表中得到原表的主键信息,为新分区表增加主键值,并删除旧表主键
create or replace procedure p_pk_009 ( p_tab in varchar2, p_deal_flag in number, p_idx_tablespace in varchar2 ) as /* 功能:从_YYYYMMDD备份表中得到原表的主键信息,为新分区表增加主键值,并删除旧表主键 难点:需要考虑联合主键的情况 */ v_sql_p_pk varchar2(4000); v_cnt number; begin select count(*) into v_cnt from user_ind_columns where index_name in (select index_name from sys.user_constraints t where table_name = upper(p_tab) || '_' || yyyymmdd and constraint_type = 'P'); if v_cnt > 0 then for i in (with t as (select index_name, table_name, column_name from user_ind_columns where index_name in (select index_name from sys.user_constraints t where table_name = upper(p_tab) || '_' || yyyymmdd and constraint_type = 'P')) select index_name, table_name, max(substr(sys_connect_by_path(column_name, ','), 2)) str from (select column_name, index_name, table_name, row_number() over(partition by index_name, table_name order by column_name) rn from t) t start with rn = 1 connect by rn = prior rn + 1 and index_name = prior index_name group by index_name, t.table_name) loop v_sql_p_pk := 'alter table ' || i.table_name || ' drop constraint ' || i.index_name || ' cascade'; p_insert_log(p_tab, 'p_deal_pk', v_sql_p_pk, '将备份出来的原表的主键删除', 9, 1); p_if_judge(v_sql_p_pk, p_deal_flag); dbms_output.put_line('操作步骤9(处理主键)-------将备份出来的原表' || i.table_name || '的' || i.str || '列的主键' || i.index_name || '删除完 毕!'); ---放在FOR循环中效率没问题,因为主键只有一个,只会循环一次 v_sql_p_pk := 'ALTER TABLE ' || p_tab || ' ADD CONSTRAINT ' || i.index_name || ' PRIMARY KEY (' || i.str || ')' || ' using index tablespace ' || p_idx_tablespace; p_insert_log(p_tab, 'p_deal_pk', v_sql_p_pk, '将新分区表的主键加上', 9, 2); p_if_judge(v_sql_p_pk, p_deal_flag); dbms_output.put_line('操作步骤9(处理主键)-------对' || p_tab || '表的' || i.str || '列增加主键' || i.index_name); ---放在FOR循环中效率没问题,因为主键只有一个,只会循环一次 end loop; else dbms_output.put_line('操作步骤9(处理主键)-------' || upper(p_tab) || '_' || yyyymmdd || '并没有主键!'); end if; end p_pk_009;
从_YYYYMMDD备份表中得到原表的外键或主键约束,为新分区表增加外键或主键,并删除旧表的外键或主键
create or replace procedure p_constraint_010 ( p_tab in varchar2, p_deal_flag in number ) as /* 功能:从_YYYYMMDD备份表中得到原表的约束,为新分区表增加约束值,并删除旧表约束 难点:需要考虑联合外键REFERENCE的情况 */ v_sql_p_constraint varchar2(4000); v_cnt number; begin select count(*) into v_cnt from user_constraints where table_name = upper(p_tab) || '_' || yyyymmdd and constraint_type = 'R'; if v_cnt > 0 then for i in (with t1 as (select /*+no_merge */ position, t.owner, t.constraint_name as constraint_name1, t.table_name as table_name1, t.column_name as column_name1 from user_cons_columns t where constraint_name in (select constraint_name from user_constraints where table_name = upper(p_tab) || '_' || yyyymmdd and constraint_type = 'R')), t2 as (select /*+no_merge */ t.position, c.constraint_name constraint_name1, t.constraint_name as constraint_name2, t.table_name as table_name2, t.column_name as column_name2, max(t.position) over(partition by c.constraint_name) max_position from user_cons_columns t, user_constraints c where c.table_name = upper(p_tab) || '_' || yyyymmdd and t.constraint_name = c.r_constraint_name and c.constraint_type = 'R'), t3 as (select t1.*, t2.constraint_name2, t2.table_name2, t2.column_name2, t2.max_position from t1, t2 where t1.constraint_name1 = t2.constraint_name1 and t1.position = t2.position) select t3.*, substr(sys_connect_by_path(column_name1, ','), 2) as fk, substr(sys_ connect_by_path(column_name2, ','), 2) as pk from t3 where position = max_position start with position = 1 connect by constraint_name1 = prior constraint_name1 and position = prior position + 1) loop v_sql_p_constraint := 'alter table ' || p_tab || '_' || yyyymmdd || ' drop constraint ' || i.constraint_name1; p_insert_log(p_tab, 'p_deal_constraint', v_sql_p_constraint, '删除原表FK外键', 10, 1); p_if_judge(v_sql_p_constraint, p_deal_flag); dbms_output.put_line('操作步骤10(处理外键)------将备份出来的' || i.table_name1 || '表的' || i.column_name1 || '列的外键' || i.constraint_name1 || '删除完毕!'); v_sql_p_constraint := 'alter table ' || p_tab || ' add constraint ' || i.constraint_ name1 || ' foreign key ( ' || i.fk || ') references ' || i.table_name2 || ' (' || i.pk || ' )'; p_insert_log(p_tab, 'p_deal_constraint', v_sql_p_constraint, '将新分区表的外键加上', 10, 2); p_if_judge(v_sql_p_constraint, p_deal_flag); dbms_output.put_line('操作步骤10(处理外键)------对' || p_tab || '表的' || i.column_ name1 || '列增加外键' || i.constraint_name1); end loop; else dbms_output.put_line('操作步骤10(处理外键)------' || upper(p_tab) || '_' || yyyymmdd || '并没有外键!'); end if; end p_constraint_010;