Covering indexes
A covering index is an index that includes all the columns required by a query, including columns that would typically not be a part of an index. This is done by using the INCLUDE keyword to list the columns you want to include.
A covering index is an efficient way to perform index-only scans, where you don't need to scan the table, just the index, to satisfy the query.
Syntax
CREATE INDEX columnA_columnB_index_name ON table_name(columnA, columnB) INCLUDE (columnC);
Example
Before you start
The examples will run on any YugabyteDB universe.
To create a universe, see Set up YugabyteDB universe.
The following exercise demonstrates how to perform an index-only scan on an expression (functional) index, and further optimize the query performance using a covering index.
-
Create and insert some rows into a table
demo
with two columnsid
andusername
.CREATE TABLE IF NOT EXISTS demo (id bigint, username text);
INSERT INTO demo SELECT n,'Number'||to_hex(n) from generate_series(1,1000) n;
-
Run a select query to fetch a row with a particular username.
SELECT * FROM demo WHERE username='Number42';
id | username ----+---------- 66 | Number42 (1 row)
-
Run another select query to show how a sequential scan runs before creating an index.
EXPLAIN ANALYZE SELECT * FROM demo WHERE upper(username)='NUMBER42';
QUERY PLAN ------------------------------------------------------------------------------------------------------ Seq Scan on demo (cost=0.00..105.00 rows=1000 width=40) (actual time=15.279..15.880 rows=1 loops=1) Filter: (upper(username) = 'NUMBER42'::text) Rows Removed by Filter: 999 Planning Time: 0.075 ms Execution Time: 15.968 ms Peak Memory Usage: 0 kB (6 rows)
-
Optimize the SELECT query by creating an expression index as follows:
CREATE INDEX demo_upper ON demo( (upper(username)) );
EXPLAIN ANALYZE SELECT upper(username) FROM demo WHERE upper(username)='NUMBER42';
QUERY PLAN ------------------------------------------------------------------------------------------------------------------- Index Scan using demo_upper on demo (cost=0.00..5.28 rows=10 width=32) (actual time=1.939..1.942 rows=1 loops=1) Index Cond: (upper(username) = 'NUMBER42'::text) Planning Time: 7.289 ms Execution Time: 2.052 ms Peak Memory Usage: 8 kB (5 rows)
Using an expression index enables faster access to the rows requested in the query. The problem is that the query planner just takes the expression, sees that there's an index on it, and knows that you'll select the
username
column and apply a function to it. It then thinks it needs theusername
column without realizing it already has the value with the function applied. In this case, an index-only scan covering the column to the index can optimize the query performance. -
Create a covering index by specifying the username column in the INCLUDE clause.
For simplicity, the
username
column is used with the INCLUDE keyword to create the covering index. Generally, a covering index allows you to perform an index-only scan if the query select list matches the columns that are included in the index and the additional columns added using the INCLUDE keyword.Ideally, specify columns that are updated frequently in the INCLUDE clause. For other cases, it is probably faster to index all the key columns.
CREATE INDEX demo_upper_covering ON demo( (upper(username))) INCLUDE (username);
EXPLAIN ANALYZE SELECT upper(username) FROM demo WHERE upper(username)='NUMBER42';
QUERY PLAN --------------------------------------------------------------------------------------------------------------------- Index Only Scan using demo_upper_covering on demo (cost=0.00..5.18 rows=10 width=32) (actual time=1.650..1.653 rows=1 loops=1) Index Cond: ((upper(username)) = 'NUMBER42'::text) Heap Fetches: 0 Planning Time: 5.258 ms Execution Time: 1.736 ms Peak Memory Usage: 8 kB (6 rows)
Learn more
- Explore the Benefits of an Index-only scan in depth with a real world example.