In the previous blog post, I had demonstrated a Global (Non-Partitioned) Index and a Local (Partitioned) Index. A Global Index itself may be partitioned. A Local Index is actually Equi-Partitioned with the Tablee.
Let me demonstrate a Global Partitioned Index on the same table now.
Given the SALES_DATA table which is partitioned by SALE_DATE, I might want to support queries by CUSTOMER_ID. What if I have millions of customers? My CUSTOMERS table might be partitioned by CUSTOMER_ID. Can I have an Index on SALES_DATA that is also partitioned by CUSTOMER_ID ?
This is a Global Partitioned Index which is prefixed on a left prefix (CUSTOMER_ID) of the Index Column(s). Oracle does not support Global Partitioned Indexes that are non-prefixed. Let me demonstrated the difference between the two.
My first attempt above is valid because the Index is partitioned on CUSTOMER_ID although it is defined on (CUSTOMER_ID, SALE_DATE). Thus, it is partitioned on a left prefix. The second index attempt fails because the Partition Key (CUSTOMER_ID) is not a left prefix (part of) the Index key (SALE_DATE alone).
Remember : The Index I've created is partitioned on CUSTOMER_ID while the Table itself is partitioned on SALE_DATE.
This Index is a Global Index so any Partition Maintenance operation (TRUNCATE, DROP, SPLIT, MERGE) may result in the *entire* Index becoming UNUSABLE unless the UPDATE INDEXES clause is used in the Partition Maintenance DDL statement. This was demonstrated with the Primary Key index being a Global Index in my earlier blog post.
What about Bitmap Indexes ? Can a Bitmap Index on a partitioned Table be either Global or Local ? Oracle does not support Global Bitmap Indexes. A Bitmap Index on a Partitioned Table has to be Local.
As demonstrated earlier, a Local Index automatically inherits the Partition Name and Tablespace Name from the Table Partitions.
To summarise, these are the Indexes on my SALES_DATA table :
Remember that for a Partitioned Index (either Global or Local), the Tablespace Name and Status are attributes at the Partition definition level not at the Index definition.
Let me demonstrate a Global Partitioned Index on the same table now.
Given the SALES_DATA table which is partitioned by SALE_DATE, I might want to support queries by CUSTOMER_ID. What if I have millions of customers? My CUSTOMERS table might be partitioned by CUSTOMER_ID. Can I have an Index on SALES_DATA that is also partitioned by CUSTOMER_ID ?
SQL> l 1 create index sales_data_glbl_part_ndx_2 2 on sales_data (customer_id) 3 global 4 partition by range (customer_id) 5 ( 6 partition p_1 values less than (501000) tablespace new_indexes, 7 partition p_2 values less than (1000001) tablespace new_indexes, 8 partition p_3 values less than (1500001) tablespace new_indexes, 9 partition p_4 values less than (maxvalue) 10* ) SQL> / Index created. SQL>
This is a Global Partitioned Index which is prefixed on a left prefix (CUSTOMER_ID) of the Index Column(s). Oracle does not support Global Partitioned Indexes that are non-prefixed. Let me demonstrated the difference between the two.
SQL> drop index sales_data_glbl_part_ndx_2; Index dropped. SQL> SQL> l 1 create index sales_data_glbl_part_ndx_2 2 on sales_data (customer_id, sale_date) 3 global 4 partition by range (customer_id) 5 ( 6 partition p_1 values less than (501000) tablespace new_indexes, 7 partition p_2 values less than (1000001) tablespace new_indexes, 8 partition p_3 values less than (1500001) tablespace new_indexes, 9 partition p_4 values less than (maxvalue) 10* ) SQL> / Index created. SQL> SQL> SQL> l 1 create index sales_data_glbl_part_ndx_3 2 on sales_data (sale_date) 3 global 4 partition by range (customer_id) 5 ( 6 partition p_1 values less than (501000) tablespace new_indexes, 7 partition p_2 values less than (1000001) tablespace new_indexes, 8 partition p_3 values less than (1500001) tablespace new_indexes, 9 partition p_4 values less than (maxvalue) 10* ) SQL> / partition by range (customer_id) * ERROR at line 4: ORA-14038: GLOBAL partitioned index must be prefixed SQL> SQL> !oerr ora 14038 14038, 00000, "GLOBAL partitioned index must be prefixed" // *Cause: User attempted to create a GLOBAL non-prefixed partitioned index // which is illegal // *Action: If the user, indeed, desired to create a non-prefixed // index, it must be created as LOCAL; otherwise, correct the list // of key and/or partitioning columns to ensure that the index is // prefixed SQL>
My first attempt above is valid because the Index is partitioned on CUSTOMER_ID although it is defined on (CUSTOMER_ID, SALE_DATE). Thus, it is partitioned on a left prefix. The second index attempt fails because the Partition Key (CUSTOMER_ID) is not a left prefix (part of) the Index key (SALE_DATE alone).
Remember : The Index I've created is partitioned on CUSTOMER_ID while the Table itself is partitioned on SALE_DATE.
This Index is a Global Index so any Partition Maintenance operation (TRUNCATE, DROP, SPLIT, MERGE) may result in the *entire* Index becoming UNUSABLE unless the UPDATE INDEXES clause is used in the Partition Maintenance DDL statement. This was demonstrated with the Primary Key index being a Global Index in my earlier blog post.
What about Bitmap Indexes ? Can a Bitmap Index on a partitioned Table be either Global or Local ? Oracle does not support Global Bitmap Indexes. A Bitmap Index on a Partitioned Table has to be Local.
SQL> create bitmap index sales_data_glbl_ndx_4 2 on sales_data(product_id) 3 / on sales_data(product_id) * ERROR at line 2: ORA-25122: Only LOCAL bitmap indexes are permitted on partitioned tables SQL> !oerr ora 25122 25122, 00000, "Only LOCAL bitmap indexes are permitted on partitioned tables" // *Cause: An attempt was made to create a global bitmap index on a partioned // table. // *Action: create a local bitmap index instead. SQL> SQL> create bitmap index sales_data_lcl_ndx_2 2 on sales_data(product_id) local 3 / Index created. SQL> SQL> select partition_name, tablespace_name 2 from user_tab_partitions 3 where table_name = 'SALES_DATA' 4 order by partition_position 5 / PARTITION_NAME TABLESPACE_NAME ------------------------------ ------------------------------ P_2018 TBS_YEAR_2018 P_2019 TBS_YEAR_2019 P_2020 TBS_YEAR_2020 P_MAXVALUE USERS SQL> select partition_name, tablespace_name, status 2 from user_ind_partitions 3 where index_name = 'SALES_DATA_LCL_NDX_2' 4 order by partition_position 5 / PARTITION_NAME TABLESPACE_NAME STATUS ------------------------------ ------------------------------ -------- P_2018 TBS_YEAR_2018 USABLE P_2019 TBS_YEAR_2019 USABLE P_2020 TBS_YEAR_2020 USABLE P_MAXVALUE USERS USABLE SQL>
As demonstrated earlier, a Local Index automatically inherits the Partition Name and Tablespace Name from the Table Partitions.
To summarise, these are the Indexes on my SALES_DATA table :
SQL> l 1 select index_name, index_type, partitioned, tablespace_name, status 2 from user_indexes 3* where table_name = 'SALES_DATA' SQL> / INDEX_NAME INDEX_TYPE PAR ------------------------------ --------------------------- --- TABLESPACE_NAME STATUS ------------------------------ -------- SYS_C0017514 NORMAL NO HEMANT VALID SALES_DATA_LCL_NDX_1 NORMAL YES N/A SALES_DATA_GLBL_PART_NDX_2 NORMAL YES N/A SALES_DATA_LCL_NDX_2 BITMAP YES N/A SQL>
Remember that for a Partitioned Index (either Global or Local), the Tablespace Name and Status are attributes at the Partition definition level not at the Index definition.
No comments:
Post a Comment