There are different methods to partition an existing table. Here we will create a non partitioned table(PRODUCT) and convert it to a partitioned table using EXCHANGE option.
EXAMPLE:
Create a normal table with index and constraints and insert some data:
conn oranet/oranet Connected. CREATE TABLE PRODUCT ( id NUMBER, code VARCHAR2(10), description VARCHAR2(50), created_date DATE ); 2 3 4 5 6 Table created. ALTER TABLE PRODUCT ADD ( CONSTRAINT PRODUCT_PK PRIMARY KEY (id) ); 2 3 Table altered. CREATE INDEX created_date_ind ON PRODUCT(created_date); Index created. INSERT INTO PRODUCT VALUES (1, 'ADAM', 'BOB', TO_DATE('05-JUN-2005', 'DD-MON-YYYY')); 1 row created. INSERT INTO PRODUCT VALUES (2, 'ADAM', 'BOB', TO_DATE('05-JUN-2005', 'DD-MON-YYYY')); 1 row created. INSERT INTO PRODUCT VALUES (4, 'ADAM', 'BOB', TO_DATE('05-AUG-2005', 'DD-MON-YYYY')); 1 row created. COMMIT; Commit complete. select table_name,partitioned from dba_tabLES where table_name='PRODUCT'; TABLE_NAME PAR ------------------------------ --- PRODUCT NO
Now create an empty interim partitioned table with similar structure as that of old table, which we will use for exchange the data from PRODUCT table. Make sure to define the one partition with maxvalue.
CREATE TABLE PRODUCT_TEMP ( id NUMBER , code VARCHAR2(10), description VARCHAR2(50), created_date DATE ) PARTITION BY RANGE (created_date) (PARTITION partitioned_2006 VALUES LESS THAN (MAXVALUE)); 2 3 4 5 6 7 8 Table created. ALTER TABLE PRODUCT_TEMP ADD ( CONSTRAINT PRODUCT_TEMP_pk PRIMARY KEY (id)); 2 Table altered. CREATE INDEX PRODUCT_TEMP_ind ON PRODUCT_TEMP (created_date) LOCAL; Index created.
Now we will use the EXCHANGE PARTITION option , which will move the data from non-partitioned table(PRODUCT) to partitioned table(PRODUCT_TEMP).
ALTER TABLE PRODUCT_TEMP EXCHANGE PARTITION partitioned_2006 WITH TABLE PRODUCT WITHOUT VALIDATION UPDATE GLOBAL INDEXES; 2 3 4 5 Table altered. select count(*) from PRODUCT_TEMP; COUNT(*) ---------- 3 select count(*) from PRODUCT; COUNT(*) ---------- 0
Once this is complete we can drop the old table and rename the new table and all it’s constraints.
drop table PRODUCT; Table dropped. alter table PRODUCT_TEMP rename to PRODUCT; Table altered. alter table PRODUCT rename constraint PRODUCT_TEMP_PK to PRODUCT_PK; Table altered. ALTER INDEX PRODUCT_TEMP_PK rename to PRODUCT_PK; Index altered. ALTER INDEX PRODUCT_TEMP_IND rename to created_date_ind; Index altered.
Lets split the existing partition.
select table_name,partitioned from dba_tabLES where table_name='PRODUCT'; TABLE_NAME PAR ------------------------------ --- PRODUCT YES Select partition_name, high_value from user_tab_partitions where table_name = 'PRODUCT'; PARTITION_NAME HIGH_VALUE ------------------------------ ------------------------------------- PARTITIONED_2006 MAXVALUE ALTER TABLE PRODUCT SPLIT partition partitioned_2006 AT (TO_DATE('30-JUN-2005','DD-MON-YYYY')) INTO (PARTITION partitioned_JUN2005, PARTITION partitioned_2006) UPDATE GLOBAL INDEXES; 2 3 4 5 Table altered. Select partition_name, high_value from user_tab_partitions where table_name = 'PRODUCT'; PARTITION_NAME HIGH_VALUE ------------------------------ ----------------------------------------------------------------------------------------- PARTITIONED_2006 MAXVALUE PARTITIONED_JUN2005 TO_DATE(' 2005-06-30 00:00:00', 'SYYYY-MM-DD HH24:MI:SS', 'NLS_CALENDAR=GREGORIAN') ALTER TABLE PRODUCT SPLIT partition partitioned_2006 AT (TO_DATE('30-JUL-2005','DD-MON-YYYY')) INTO (PARTITION partitioned_JULY2005, PARTITION partitioned_2006) UPDATE GLOBAL INDEXES; 2 3 4 5 Table altered. Select partition_name, high_value from user_tab_partitions where table_name = 'PRODUCT'; PARTITION_NAME HIGH_VALUE ------------------------------ ----------------------------------------------------------------------------------------- PARTITIONED_2006 MAXVALUE PARTITIONED_JULY2005 TO_DATE(' 2005-07-30 00:00:00', 'SYYYY-MM-DD HH24:MI:SS', 'NLS_CALENDAR=GREGORIAN') PARTITIONED_JUN2005 TO_DATE(' 2005-06-30 00:00:00', 'SYYYY-MM-DD HH24:MI:SS', 'NLS_CALENDAR=GREGORIAN') EXEC DBMS_STATS.gather_table_stats('ORANET', 'PRODUCT', cascade = > TRUE); PL/SQL procedure successfully completed. Select partition_name, high_value,num_rowS from user_tab_partitions where table_name = 'PRODUCT'; PARTITION_NAME HIGH_VALUE NUM_ROWS ------------------------------ ----------------------------------------------------------------------------------------- ---------- PARTITIONED_2006 MAXVALUE 1 PARTITIONED_JULY2005 TO_DATE(' 2005-07-30 00:00:00', 'SYYYY-MM-DD HH24:MI:SS', 'NLS_CALENDAR=GREGORIAN') 0 PARTITIONED_JUN2005 TO_DATE(' 2005-06-30 00:00:00', 'SYYYY-MM-DD HH24:MI:SS', 'NLS_CALENDAR=GREGORIAN') 2
There are other methods like dbms_redefinition and exp imp method to partition and existing table. I will upload them very soon.
For any issue or if you need more information please contact [email protected]
hi
I have few question about the exchange example, i have a non-partitioned table with 2GB data inside it and want to exchange with a partitioned table (same column with base table). but the partitioned table has more than one partition. However, how can i exchange my tables together?
What is the correct command and what comes up with the data? Will the data be in the partitions?
Dear,
Exchange method is used to convert a non-partitioned table to a partitioned table. But here your requirement is to exchange between a partitioned table and non-partitioned table, where both contains data. I am afraid, exchange method wont be of helpful.
So best method is you do exchange on these two separately.
Regards
Admin
Thank you,
both tables doesn’t contain data, just non-partitioned table has data. In your example, the partitioned table just has one partition, so the EXCHANGE PARTITION command is simple, but my partitioned table has more than one partition. I don’t know how can i exchange these tables.
In your example, you exchange PRODUCT with partitioned_2006 of PRODUCT_TEMP.
My partitioned table have at least 27 partitions (partitioned by list method) and i want to move data in all partitions.
please help me how can i solve this issue. what is your solution?
Is this command correct?
CREATE TABLE FOR EXCHANGE WITH
PARTITION BY HASH (partition_key)
(
PARTITION 16
);
CREATE TABLE table name FOR EXCHANGE WITH table name
PARTITION BY HASH (partition_key)
(
PARTITION 16
);
Hi Dear,
Steps are same. You can have more than 1 partition also.
HASH PARTITION:
———————
create table emp2 (
empno number(4),
ename varchar2(30),
sal number
)
partition by hash(empno) (
partition e1 tablespace emp1,
partition e2 tablespace emp2,
partition e3 tablespace emp3,
partition e4 tablespace emp4
);
For range partition:
————————-
CREATE TABLE PRODUCT_TEMP (
id NUMBER ,
code VARCHAR2(10),
description VARCHAR2(50),
created_date DATE
)
PARTITION BY RANGE (created_date)
(PARTITION partitioned_2006 VALUES LESS THAN TO_DATE(‘2007-01-01’, ‘YYYY-MM-DD’)),
(PARTITION partitioned_2007 VALUES LESS THAN TO_DATE(‘2008-01-01’, ‘YYYY-MM-DD’)),
(PARTITION partitioned_2008 VALUES LESS THAN (MAXVALUE));
ALTER TABLE PRODUCT ADD (
CONSTRAINT PRODUCT_pk PRIMARY KEY (id)); — if no this, exchange partition will prompt “ORA-14097: column type or size mismatch in ALTER TABLE EXCHANGE PARTITION”