Oracle ORDER BY Index | How to Perform? | 6 Examples

Asiri Gunasena

Published:

SQL

oracle order by index

Oracle order by index is a good option to sort data by improving performance. Sorting is used when you want to order the data in a specific manner but if the data load is very high then sorting takes a big portion of time from full query execution time. there you need to improve the order by performance in the query.

In this article, we discuss improving the performance by using Oracle order by index. If you wrote a normal query without an index optimizer has to read the full table to sort out the dataset. Here we are looking to reduce time by adding an index to sorting columns. It ignores the full table scan and uses an index to show the results directly. Because indexes are already in an ordered way.

Here I would like to explain 5 simple examples of creating an index for ordering with corresponding SQL code and how to create an oracle order by index in a correct way. Because sometimes we have an index but not in an optimal way. Then we have improvement but there can be more.

Oracle ORDER BY Index Examples

Example 1: Basic Index on ORDER BY Column

This is the most basic example of an index. Suppose you have a table called Employee and column as Salary then you would like to sort the employees by salary. Then you can create an index on the salary column to improve the performance.

--Query to fetch data and ordered by salary 
SELECT * FROM sales ORDER BY order_date; 

-- Create an index on the order_date column 
CREATE INDEX idx_order_date ON sales(order_date); 
-- 

I created the B-TREE index because there are big variations can be there in the column. But you have to stick to a situation to create an index. Because there are index types like DOMAIN, BITMAP, and more. Those are working well according to the data set. Therefore creating an index in an oracle not only improves the performance but analyzes the dataset and creates an appropriate type of index.

Example 2: Covering Index

The covering index in Oracle includes all the columns required for a query. This is not valid if you use a lot of columns in the query. Because creating columns with including a lot of columns reduces the performance when modifying data. Therefore I think this is better for small queries.

-- Query  to retrieve data
SELECT order_id, order_date
FROM orders
WHERE customer_id = 123
ORDER BY order_date;

-- Create index for order_date including Customer id and orader_id
CREATE INDEX idx_covering_orders ON orders(order_id, order_date ,customer_id);
--

Covering Index fulfills the query without accessing the database table. all the data in the index therefore no need to go and fetch data from a database table. This is reducing the I/O call for the database. Create in in most variation available column first. That also helps to improve the performance

Example 3: Reverse Index for Descending Order

This is a simple concept of index. You may know indexes are in order. But you prefer to load data in descending order and create an index also in descending order. Then query can use index order as it is. If you have an index on ascending order for other purposes then no need to create a new one for descending order because Oracle uses it but creating two indexes in both orders may not help much for performance.

-- Create a DESC index on the order_date 
CREATE INDEX idx_reverse_order_date ON sales(order_date DESC); 

-- Query fetch in descending order 
SELECT * FROM sales ORDER BY order_date DESC; 
--

Example 4: Composite Index

If you want to order with multiple columns create an Oracle order by index in the same order in the query.

-- Create a composite index 
CREATE INDEX idx_order ON purchase_order_table(order_date, customer_id); 

-- Query to fetch and ordered by order date and customer ID 
SELECT * FROM purchase_order_table ORDER BY order_date, customer_id
--

Example 5: Functional Index

The function base index is one of the index types you can use when ordering data.

-- Create a functional index 
CREATE INDEX idx_order ON purchase_order_table(round(cost)); 
-- Query to fetch data
SELECT * FROM purchase_order_table ORDER BY round (cost);

NOTE: IF you want to order by using JOIN columns then the index is not used by Oracle.

Order by Index Complete Analysis -IMPORTANT

I will do the complete analysis on creating Oracle order by index and do the performance improvement. If you don’t have any index on order by related columns, then Oracle does the full table scan for sorting. I insert 1 million data into the sample table and let’s improve the performance in a query by adding an index.

Sample data

--
CREATE TABLE purchase_order_table(
    order_id VARCHAR2(50) ,
    order_date DATE ,
    need_date DATE ,
    quantity NUMBER,
    PRIMARY KEY(order_id)
);
declare 
  -- insert 100000 records into database
  i integer:=0;
begin
LOOP
   i:= i+1;
   INSERT INTO purchase_order_table (order_id, order_date,need_date,quantity) 
   VALUES ('Po-'||i,Sysdate - round(dbms_random.value(1,400),0),Sysdate + round(dbms_random.value(1,400),0),round(dbms_random.value(1,1000),0));
   EXIT WHEN i > 999999;
END LOOP;
end;

Sample Query

SELECT order_id, quantity, order_date, need_date
  FROM purchase_order_table
 WHERE need_date < sysdate+5
 ORDER BY need_date, order_date;
oracle order by index

In this demo query, you can see need_date and the order_date in order by clause and need_date in the where clause as well. Let’s try to Explain Plan for this first. According to the Explain Plan oracle does the full table scan for filtering and then does the sorting again.

oracle order by index

I will create an index for need_date because of need_date in my where clause. Hope the query will improve the performance by using the index.

--
CREATE INDEX idx_need_date ON purchase_order_table (need_date);
--
oracle order by index

Index created and then let’s explain what looks like. It Is reduced rgt? COST was also reduced and clearly now Oracle does the INDEX RANGE SCAN and uses an index for it. BUT still, a problem here is SORT ORDER BY took considerable time. I hope this is still able to improve. So, I will modify the index by adding order_date into an index. Let’s look at any time reduction for it.

--
CREATE INDEX index_date ON purchase_order_table (need_date, order_date)
--
Oracle order by index
Oracle order by index

Now Explain Plan and say I do not need SORT ORDER BY operation anymore. So, Oracle has done the sort by indexing itself. So, we got some gain from it, and time reduced in our query. This is how we can use the index for sorting and ordering data by multiple columns. Because of index are already ordered then oracle accesses the record by ROWID.

Hope you that is enough. YES if you are writing usual queries and those are not used frequently this is enough I guess. But according to the situation, you can decide where you need satisfied. Now I will create a CONVERING INDEX. This is a need to create when you have queries that are running frequently. Consider about index range and memory size for the index.

--
CREATE INDEX index_date ON purchase_order_table (need_date, order_date,order_id,quantity)
--

You can see I include all the columns in the index. According to my earlier Explain Plan Out of COST, most of the time took for I/O COST. Therefore, I want to reduce that cost. It is because of remaining columns need to fetch from a table. That I/O time can be reduced by adding those columns to the index.

oracle order by index

Now you can see COST reduced and other performance matters like Bytes, and CPU COST are reduced due to our new index. BUT we do not recommend this to be done every time because record INSERT and UPDATE may take extra time to update due to this type of composite index.

This is Wrong Create Order by Index.

As we discussed composite index can be created any way this is the wrong way of creating an index for our example because this does not get maximum performance by the index

--
create INDEX index_date ON purchase_order_table (order_id,quantity,order_date,need_date)
--

It is still scanned through the index. you can see it in the Explain Plan by saying INDEX FULL SCAN but that is not what we expected. So mindful when you create an index with the correct column order.

oracle order by index
Case vs Decode in Oracle Performance
Case vs Decode in Oracle Performance

Conclusion

oracle order by index is a great way of performance improvement. An index can be created in different ways like B-TREE, DOMAIN, BITMAP, Function base or there are a lot. But create an index by analyzing the query and the data load in the table. Also, prefer to improve the query according to the situation where you going to use this.

Sometimes index perform well on fetching but Modification has some problems. So balance of sort of operation in the table and do the performance. There are a lot of performance-related articles on this site. Read and comment on our improvement as well.

Resources

https://use-the-index-luke.com/sql/sorting-grouping/indexed-order-by

https://docs.oracle.com/cd/E14004_01/books/PerformTun/PerformTunCustConfig14.html

Categories SQL
ennicode

Address: 89/1 Rabbegamuwa, Handessa, Kandy, Central, Sri Lanka

Email to: Primary: [email protected]

Services

E-Learning

Company Websites

Support and Configuration work

Banners, Covers, and Post

Web Development & Configurations

Content Writing and Marketing

Contact

Ennicode