views:

139

answers:

5

I have a table with multiple indexes, several of which duplicate the same columns:

Index 1 columns: X, B, C, D
Index 2 columns: Y, B, C, D
Index 3 columns: Z, B, C, D

I'm not very knowledgeable on indexing in practice, so I'm wondering if somebody can explain why X, Y and Z were paired with these same columns. B is an effective date. C is a semi-unique key ID for this table for a specific effective date B. D is a sequence that identifies the priority of this record for the identifier C.

Why not just create 6 indexes, one for each X, Y, Z, B, C, D?

I want to add an index to another column T, but in some contexts I'll only be querying on T alone while in others I will also be specifying the B, C and D columns... so should I create just one index like above or should I create one for T and one for (T, B, C, D)?

I've not had as much luck as expected when googling for comprehensive coverage of indexing. Any resources where I can get a through explanation and lots of examples of B-tree indexing?

+3  A: 

One reason for having B, C and D in those indexes might be to have a covering index for frequently used queries. You will have a covering index when the index itself contains all the required data fields for a particular query.

A covering index can dramatically speed up data retrieval, since only the index pages, not the data pages, will be used to retrieve the data.

Below is an example query where index 1 would be a covering index:

SELECT B, C, D FROM table WHERE X = '10'
Daniel Vassallo
+6  A: 

The rule with indexing is that an index can be used to filter on any list of columns that constitute a prefix of the columns used for that index.

In other words, we can use Index 1 when we filter on X and B, or X, B and C, or just X, or all four.

However, we cannot use the index to filter "in the middle". This is because indexes work not entirely unlike concatenating the values of those columns for each row, and sorting the result. If we know what the thing we're looking for begins with, we can figure out where in the index to look - just like when doing binary search.

That's why a single index is no good: if we need to filter on B, C, D, and one of X, Y and Z, we need three indexes; X, Y is no good as an index for just filtering on Y, because the prefix of the values we're looking for - the X - is not known.

As Daniel mentioned, a covering index is a possible explanation for repeating B, C, and D: even if D is never filtered on, it may be the case that we need exactly the columns which you see in your indexes, and we can then just read the columns from the index instead of just using the index to locate the row.

Michael Madsen
Actually, you can use an index to drive access for columns "in the middle" -- Oracle supports skip scanning, so an index on `X, B, C` could be used for an equality condition on B and C where X is not present. It's a reason to consider index key compression and leading the index with the least selective columns.
Adam Musch
+1  A: 

You should create it in (T, B, C, D).

Let's say you have two fields with an index in a table: A and B. When you create a separate index on each one of the columns, and have a query such as:

SELECT * FROM table WHERE A = 10 AND B = 20

What happens is either:

1) The DB creates two intermediate result-sets, one with rows where A = 10, and another one with rows where B = 20. It then has to merge these two result-sets into one (and also check for duplicate rows).

2) The DB creates one result-set with rows where A = 10. It then has to go manually through all of the rows in this intermediate result-set and check in each one where B = 10.

However when you know that index B depends on index A, and your query uses A before B, you can create one index for both of the columns: (A, B)

What this means that now the DB will first find all rows where A = 10, but because B is part of the same index, it can use the same index information to filter the result-set into rows where B is also 20. It doesn't have to make two intermediate result-sets + merge them, or only use one of the indexes and do manual scan for the other.

There might be other ways that the DB deals with these situations as well, it largely depends on an implementation.

reko_t
+1  A: 

The indexes in the form (X, B, C, D) can be used to optimize queries like:

... WHERE X rel sthg (possibly ORDER BY B, C, D)
... WHERE X = sthg AND B rel sthg (possibly ORDER BY C, D)
... WHERE X = sthf AND B = sthg AND C rel sthg (possibly ORDER BY D)

etc. where rel are arbitrary relation operators (<, >, =, <=, >=) and sthg are values or expressions. Especially the second two, and the sorting variants wouldn't be optimized by the "single column indexes variant".

OTOH, it cannot optimize a query

... WHERE B = sthg

because it starts in the middle of the index; here, the single column index would work.

jpalecek
A: 

For a resource where you can get a through explanation and lots of examples regarding indexes on Oracle (and any other Oracle-related issue), you should visit and bookmark askTom.

JorgeLarre