Commit e6dd664d authored by Tom Lane's avatar Tom Lane

Docs: create some user-facing documentation about index-only scans.

We didn't have any real user documentation about how index-only scans
work or how to design indexes to exploit them.  Remedy that.
Per gripe from David Johnston.
parent 6f69b963
...@@ -3406,9 +3406,6 @@ include_dir 'conf.d' ...@@ -3406,9 +3406,6 @@ include_dir 'conf.d'
<varlistentry id="guc-enable-indexonlyscan" xreflabel="enable_indexonlyscan"> <varlistentry id="guc-enable-indexonlyscan" xreflabel="enable_indexonlyscan">
<term><varname>enable_indexonlyscan</varname> (<type>boolean</type>) <term><varname>enable_indexonlyscan</varname> (<type>boolean</type>)
<indexterm>
<primary>index-only scan</primary>
</indexterm>
<indexterm> <indexterm>
<primary><varname>enable_indexonlyscan</> configuration parameter</primary> <primary><varname>enable_indexonlyscan</> configuration parameter</primary>
</indexterm> </indexterm>
...@@ -3416,7 +3413,8 @@ include_dir 'conf.d' ...@@ -3416,7 +3413,8 @@ include_dir 'conf.d'
<listitem> <listitem>
<para> <para>
Enables or disables the query planner's use of index-only-scan plan Enables or disables the query planner's use of index-only-scan plan
types. The default is <literal>on</>. types (see <xref linkend="indexes-index-only-scans">).
The default is <literal>on</>.
</para> </para>
</listitem> </listitem>
</varlistentry> </varlistentry>
......
...@@ -354,10 +354,11 @@ amvacuumcleanup (IndexVacuumInfo *info, ...@@ -354,10 +354,11 @@ amvacuumcleanup (IndexVacuumInfo *info,
bool bool
amcanreturn (Relation indexRelation, int attno); amcanreturn (Relation indexRelation, int attno);
</programlisting> </programlisting>
Check whether the index can support <firstterm>index-only scans</> on the Check whether the index can support <link
given column, by returning the indexed column values for an index entry in linkend="indexes-index-only-scans"><firstterm>index-only scans</></link> on
the form of an <structname>IndexTuple</structname>. The attribute number the given column, by returning the indexed column values for an index entry
is 1-based, i.e. the first columns attno is 1. Returns TRUE if supported, in the form of an <structname>IndexTuple</structname>. The attribute number
is 1-based, i.e. the first column's attno is 1. Returns TRUE if supported,
else FALSE. If the access method does not support index-only scans at all, else FALSE. If the access method does not support index-only scans at all,
the <structfield>amcanreturn</> field in its <structname>IndexAmRoutine</> the <structfield>amcanreturn</> field in its <structname>IndexAmRoutine</>
struct can be set to NULL. struct can be set to NULL.
...@@ -489,8 +490,8 @@ amgettuple (IndexScanDesc scan, ...@@ -489,8 +490,8 @@ amgettuple (IndexScanDesc scan,
</para> </para>
<para> <para>
If the index supports index-only scans (i.e., If the index supports <link linkend="indexes-index-only-scans">index-only
<function>amcanreturn</function> returns TRUE for it), scans</link> (i.e., <function>amcanreturn</function> returns TRUE for it),
then on success the AM must also check then on success the AM must also check
<literal>scan-&gt;xs_want_itup</>, and if that is true it must return <literal>scan-&gt;xs_want_itup</>, and if that is true it must return
the original indexed data for the index entry, in the form of an the original indexed data for the index entry, in the form of an
...@@ -700,9 +701,10 @@ amrestrpos (IndexScanDesc scan); ...@@ -700,9 +701,10 @@ amrestrpos (IndexScanDesc scan);
<para> <para>
If the index stores the original indexed data values (and not some lossy If the index stores the original indexed data values (and not some lossy
representation of them), it is useful to support index-only scans, in representation of them), it is useful to
support <link linkend="indexes-index-only-scans">index-only scans</link>, in
which the index returns the actual data not just the TID of the heap tuple. which the index returns the actual data not just the TID of the heap tuple.
This will only work if the visibility map shows that the TID is on an This will only avoid I/O if the visibility map shows that the TID is on an
all-visible page; else the heap tuple must be visited anyway to check all-visible page; else the heap tuple must be visited anyway to check
MVCC visibility. But that is no concern of the access method's. MVCC visibility. But that is no concern of the access method's.
</para> </para>
......
...@@ -302,9 +302,16 @@ SELECT * FROM places ORDER BY location <-> point '(101,456)' LIMIT 10; ...@@ -302,9 +302,16 @@ SELECT * FROM places ORDER BY location <-> point '(101,456)' LIMIT 10;
<primary>GIN</primary> <primary>GIN</primary>
<see>index</see> <see>index</see>
</indexterm> </indexterm>
GIN indexes are inverted indexes which can handle values that contain more GIN indexes are <quote>inverted indexes</> which are appropriate for
than one key, arrays for example. Like GiST and SP-GiST, GIN can support data values that contain multiple component values, such as arrays. An
many different user-defined indexing strategies and the particular inverted index contains a separate entry for each component value, and
can efficiently handle queries that test for the presence of specific
component values.
</para>
<para>
Like GiST and SP-GiST, GIN can support
many different user-defined indexing strategies, and the particular
operators with which a GIN index can be used vary depending on the operators with which a GIN index can be used vary depending on the
indexing strategy. indexing strategy.
As an example, the standard distribution of As an example, the standard distribution of
...@@ -337,16 +344,16 @@ SELECT * FROM places ORDER BY location <-> point '(101,456)' LIMIT 10; ...@@ -337,16 +344,16 @@ SELECT * FROM places ORDER BY location <-> point '(101,456)' LIMIT 10;
<primary>BRIN</primary> <primary>BRIN</primary>
<see>index</see> <see>index</see>
</indexterm> </indexterm>
BRIN indexes (a shorthand for Block Range indexes) BRIN indexes (a shorthand for Block Range INdexes) store summaries about
store summaries about the values stored in consecutive table physical block ranges. the values stored in consecutive physical block ranges of a table.
Like GiST, SP-GiST and GIN, Like GiST, SP-GiST and GIN,
BRIN can support many different indexing strategies, BRIN can support many different indexing strategies,
and the particular operators with which a BRIN index can be used and the particular operators with which a BRIN index can be used
vary depending on the indexing strategy. vary depending on the indexing strategy.
For data types that have a linear sort order, the indexed data For data types that have a linear sort order, the indexed data
corresponds to the minimum and maximum values of the corresponds to the minimum and maximum values of the
values in the column for each block range, values in the column for each block range. This supports indexed queries
which support indexed queries using these operators: using these operators:
<simplelist> <simplelist>
<member><literal>&lt;</literal></member> <member><literal>&lt;</literal></member>
...@@ -460,7 +467,8 @@ CREATE INDEX test2_mm_idx ON test2 (major, minor); ...@@ -460,7 +467,8 @@ CREATE INDEX test2_mm_idx ON test2 (major, minor);
an index on a single column is sufficient and saves space and time. an index on a single column is sufficient and saves space and time.
Indexes with more than three columns are unlikely to be helpful Indexes with more than three columns are unlikely to be helpful
unless the usage of the table is extremely stylized. See also unless the usage of the table is extremely stylized. See also
<xref linkend="indexes-bitmap-scans"> for some discussion of the <xref linkend="indexes-bitmap-scans"> and
<xref linkend="indexes-index-only-scans"> for some discussion of the
merits of different index configurations. merits of different index configurations.
</para> </para>
</sect1> </sect1>
...@@ -1140,6 +1148,183 @@ CREATE INDEX test1c_content_y_index ON test1c (content COLLATE "y"); ...@@ -1140,6 +1148,183 @@ CREATE INDEX test1c_content_y_index ON test1c (content COLLATE "y");
</sect1> </sect1>
<sect1 id="indexes-index-only-scans">
<title>Index-Only Scans</title>
<indexterm zone="indexes-index-only-scans">
<primary>index</primary>
<secondary>index-only scans</secondary>
</indexterm>
<indexterm zone="indexes-index-only-scans">
<primary>index-only scan</primary>
</indexterm>
<para>
All indexes in <productname>PostgreSQL</> are <firstterm>secondary</>
indexes, meaning that each index is stored separately from the table's
main data area (which is called the table's <firstterm>heap</>
in <productname>PostgreSQL</> terminology). This means that in an
ordinary index scan, each row retrieval requires fetching data from both
the index and the heap. Furthermore, while the index entries that match a
given indexable <literal>WHERE</> condition are usually close together in
the index, the table rows they reference might be anywhere in the heap.
The heap-access portion of an index scan thus involves a lot of random
access into the heap, which can be slow, particularly on traditional
rotating media. (As described in <xref linkend="indexes-bitmap-scans">,
bitmap scans try to alleviate this cost by doing the heap accesses in
sorted order, but that only goes so far.)
</para>
<para>
To solve this performance problem, <productname>PostgreSQL</>
supports <firstterm>index-only scans</>, which can answer queries from an
index alone without any heap access. The basic idea is to return values
directly out of each index entry instead of consulting the associated heap
entry. There are two fundamental restrictions on when this method can be
used:
<orderedlist>
<listitem>
<para>
The index type must support index-only scans. B-tree indexes always
do. GiST and SP-GiST indexes support index-only scans for some
operator classes but not others. Other index types have no support.
The underlying requirement is that the index must physically store, or
else be able to reconstruct, the original data value for each index
entry. As a counterexample, GIN indexes cannot support index-only
scans because each index entry typically holds only part of the
original data value.
</para>
</listitem>
<listitem>
<para>
The query must reference only columns stored in the index. For
example, given an index on columns <literal>x</> and <literal>y</> of a
table that also has a column <literal>z</>, these queries could use
index-only scans:
<programlisting>
SELECT x, y FROM tab WHERE x = 'key';
SELECT x FROM tab WHERE x = 'key' AND y &lt; 42;
</programlisting>
but these queries could not:
<programlisting>
SELECT x, z FROM tab WHERE x = 'key';
SELECT x FROM tab WHERE x = 'key' AND z &lt; 42;
</programlisting>
(Expression indexes and partial indexes complicate this rule,
as discussed below.)
</para>
</listitem>
</orderedlist>
</para>
<para>
If these two fundamental requirements are met, then all the data values
required by the query are available from the index, so an index-only scan
is physically possible. But there is an additional requirement for any
table scan in <productname>PostgreSQL</>: it must verify that each
retrieved row be <quote>visible</> to the query's MVCC snapshot, as
discussed in <xref linkend="mvcc">. Visibility information is not stored
in index entries, only in heap entries; so at first glance it would seem
that every row retrieval would require a heap access anyway. And this is
indeed the case, if the table row has been modified recently. However,
for seldom-changing data there is a way around this
problem. <productname>PostgreSQL</> tracks, for each page in a table's
heap, whether all rows stored in that page are old enough to be visible to
all current and future transactions. This information is stored in a bit
in the table's <firstterm>visibility map</>. An index-only scan, after
finding a candidate index entry, checks the visibility map bit for the
corresponding heap page. If it's set, the row is known visible and so the
data can be returned with no further work. If it's not set, the heap
entry must be visited to find out whether it's visible, so no performance
advantage is gained over a standard index scan. Even in the successful
case, this approach trades visibility map accesses for heap accesses; but
since the visibility map is four orders of magnitude smaller than the heap
it describes, far less physical I/O is needed to access it. In most
situations the visibility map remains cached in memory all the time.
</para>
<para>
In short, while an index-only scan is possible given the two fundamental
requirements, it will be a win only if a significant fraction of the
table's heap pages have their all-visible map bits set. But tables in
which a large fraction of the rows are unchanging are common enough to
make this type of scan very useful in practice.
</para>
<para>
To make effective use of the index-only scan feature, you might choose to
create indexes in which only the leading columns are meant to
match <literal>WHERE</> clauses, while the trailing columns
hold <quote>payload</> data to be returned by a query. For example, if
you commonly run queries like
<programlisting>
SELECT y FROM tab WHERE x = 'key';
</programlisting>
the traditional approach to speeding up such queries would be to create an
index on <literal>x</> only. However, an index on <literal>(x, y)</>
would offer the possibility of implementing this query as an index-only
scan. As previously discussed, such an index would be larger and hence
more expensive than an index on <literal>x</> alone, so this is attractive
only if the table is known to be mostly static. Note it's important that
the index be declared on <literal>(x, y)</> not <literal>(y, x)</>, as for
most index types (particularly B-trees) searches that do not constrain the
leading index columns are not very efficient.
</para>
<para>
In principle, index-only scans can be used with expression indexes.
For example, given an index on <literal>f(x)</> where <literal>x</> is a
table column, it should be possible to execute
<programlisting>
SELECT f(x) FROM tab WHERE f(x) &lt; 1;
</programlisting>
as an index-only scan; and this is very attractive if <literal>f()</> is
an expensive-to-compute function. However, <productname>PostgreSQL</>'s
planner is currently not very smart about such cases. It considers a
query to be potentially executable by index-only scan only when
all <emphasis>columns</> needed by the query are available from the index.
In this example, <literal>x</> is not needed except in the
context <literal>f(x)</>, but the planner does not notice that and
concludes that an index-only scan is not possible. If an index-only scan
seems sufficiently worthwhile, this can be worked around by declaring the
index to be on <literal>(f(x), x)</>, where the second column is not
expected to be used in practice but is just there to convince the planner
that an index-only scan is possible. An additional caveat, if the goal is
to avoid recalculating <literal>f(x)</>, is that the planner won't
necessarily match uses of <literal>f(x)</> that aren't in
indexable <literal>WHERE</> clauses to the index column. It will usually
get this right in simple queries such as shown above, but not in queries
that involve joins. These deficiencies may be remedied in future versions
of <productname>PostgreSQL</>.
</para>
<para>
Partial indexes also have interesting interactions with index-only scans.
Consider the partial index shown in <xref linkend="indexes-partial-ex3">:
<programlisting>
CREATE UNIQUE INDEX tests_success_constraint ON tests (subject, target)
WHERE success;
</programlisting>
In principle, we could do an index-only scan on this index to satisfy a
query like
<programlisting>
SELECT target FROM tests WHERE subject = 'some-subject' AND success;
</programlisting>
But there's a problem: the <literal>WHERE</> clause refers
to <literal>success</> which is not available as a result column of the
index. Nonetheless, an index-only scan is possible because the plan does
not need to recheck that part of the <literal>WHERE</> clause at runtime:
all entries found in the index necessarily have <literal>success = true</>
so this need not be explicitly checked in the
plan. <productname>PostgreSQL</> versions 9.6 and later will recognize
such cases and allow index-only scans to be generated, but older versions
will not.
</para>
</sect1>
<sect1 id="indexes-examine"> <sect1 id="indexes-examine">
<title>Examining Index Usage</title> <title>Examining Index Usage</title>
......
...@@ -102,8 +102,9 @@ ...@@ -102,8 +102,9 @@
</listitem> </listitem>
<listitem> <listitem>
<simpara>To update the visibility map, which speeds up index-only <simpara>To update the visibility map, which speeds
scans.</simpara> up <link linkend="indexes-index-only-scans">index-only
scans</link>.</simpara>
</listitem> </listitem>
<listitem> <listitem>
...@@ -363,9 +364,11 @@ ...@@ -363,9 +364,11 @@
Since <productname>PostgreSQL</productname> indexes don't contain tuple Since <productname>PostgreSQL</productname> indexes don't contain tuple
visibility information, a normal index scan fetches the heap tuple for each visibility information, a normal index scan fetches the heap tuple for each
matching index entry, to check whether it should be seen by the current matching index entry, to check whether it should be seen by the current
transaction. An <firstterm>index-only scan</>, on the other hand, checks transaction.
the visibility map first. If it's known that all tuples on the page are An <link linkend="indexes-index-only-scans"><firstterm>index-only
visible, the heap fetch can be skipped. This is most noticeable on scan</></link>, on the other hand, checks the visibility map first.
If it's known that all tuples on the page are
visible, the heap fetch can be skipped. This is most useful on
large data sets where the visibility map can prevent disk accesses. large data sets where the visibility map can prevent disk accesses.
The visibility map is vastly smaller than the heap, so it can easily be The visibility map is vastly smaller than the heap, so it can easily be
cached even when the heap is very large. cached even when the heap is very large.
......
...@@ -636,9 +636,11 @@ Note that indexes do not have VMs. ...@@ -636,9 +636,11 @@ Note that indexes do not have VMs.
The visibility map stores two bits per heap page. The first bit, if set, The visibility map stores two bits per heap page. The first bit, if set,
indicates that the page is all-visible, or in other words that the page does indicates that the page is all-visible, or in other words that the page does
not contain any tuples that need to be vacuumed. not contain any tuples that need to be vacuumed.
This information can also be used by <firstterm>index-only scans</> to answer This information can also be used
queries using only the index tuple. by <link linkend="indexes-index-only-scans"><firstterm>index-only
scans</></link> to answer queries using only the index tuple.
The second bit, if set, means that all tuples on the page have been frozen. The second bit, if set, means that all tuples on the page have been frozen.
That means that even an anti-wraparound vacuum need not revisit the page.
</para> </para>
<para> <para>
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment