mirror of
https://github.com/postgres/postgres.git
synced 2025-07-28 23:42:10 +03:00
Improve planning of btree index scans using ScalarArrayOpExpr quals.
Since we taught btree to handle ScalarArrayOpExpr quals natively (commit9e8da0f757
), the planner has always included ScalarArrayOpExpr quals in index conditions if possible. However, if the qual is for a non-first index column, this could result in an inferior plan because we can no longer take advantage of index ordering (cf. commit807a40c551
). It can be better to omit the ScalarArrayOpExpr qual from the index condition and let it be done as a filter, so that the output doesn't need to get sorted. Indeed, this is true for the query introduced as a test case by the latter commit. To fix, restructure get_index_paths and build_index_paths so that we consider paths both with and without ScalarArrayOpExpr quals in non-first index columns. Redesign the API of build_index_paths so that it reports what it found, saving useless second or third calls. Report and patch by Andrew Gierth (though rather heavily modified by me). Back-patch to 9.2 where this code was introduced, since the issue can result in significant performance regressions compared to plans produced by 9.1 and earlier.
This commit is contained in:
@ -2754,6 +2754,27 @@ ORDER BY unique1;
|
||||
42
|
||||
(3 rows)
|
||||
|
||||
explain (costs off)
|
||||
SELECT thousand, tenthous FROM tenk1
|
||||
WHERE thousand < 2 AND tenthous IN (1001,3000)
|
||||
ORDER BY thousand;
|
||||
QUERY PLAN
|
||||
-------------------------------------------------------
|
||||
Index Only Scan using tenk1_thous_tenthous on tenk1
|
||||
Index Cond: (thousand < 2)
|
||||
Filter: (tenthous = ANY ('{1001,3000}'::integer[]))
|
||||
(3 rows)
|
||||
|
||||
SELECT thousand, tenthous FROM tenk1
|
||||
WHERE thousand < 2 AND tenthous IN (1001,3000)
|
||||
ORDER BY thousand;
|
||||
thousand | tenthous
|
||||
----------+----------
|
||||
0 | 3000
|
||||
1 | 1001
|
||||
(2 rows)
|
||||
|
||||
SET enable_indexonlyscan = OFF;
|
||||
explain (costs off)
|
||||
SELECT thousand, tenthous FROM tenk1
|
||||
WHERE thousand < 2 AND tenthous IN (1001,3000)
|
||||
@ -2762,7 +2783,7 @@ ORDER BY thousand;
|
||||
--------------------------------------------------------------------------------------
|
||||
Sort
|
||||
Sort Key: thousand
|
||||
-> Index Only Scan using tenk1_thous_tenthous on tenk1
|
||||
-> Index Scan using tenk1_thous_tenthous on tenk1
|
||||
Index Cond: ((thousand < 2) AND (tenthous = ANY ('{1001,3000}'::integer[])))
|
||||
(4 rows)
|
||||
|
||||
@ -2775,6 +2796,7 @@ ORDER BY thousand;
|
||||
1 | 1001
|
||||
(2 rows)
|
||||
|
||||
RESET enable_indexscan;
|
||||
--
|
||||
-- Check elimination of constant-NULL subexpressions
|
||||
--
|
||||
|
@ -932,6 +932,19 @@ SELECT thousand, tenthous FROM tenk1
|
||||
WHERE thousand < 2 AND tenthous IN (1001,3000)
|
||||
ORDER BY thousand;
|
||||
|
||||
SET enable_indexonlyscan = OFF;
|
||||
|
||||
explain (costs off)
|
||||
SELECT thousand, tenthous FROM tenk1
|
||||
WHERE thousand < 2 AND tenthous IN (1001,3000)
|
||||
ORDER BY thousand;
|
||||
|
||||
SELECT thousand, tenthous FROM tenk1
|
||||
WHERE thousand < 2 AND tenthous IN (1001,3000)
|
||||
ORDER BY thousand;
|
||||
|
||||
RESET enable_indexscan;
|
||||
|
||||
--
|
||||
-- Check elimination of constant-NULL subexpressions
|
||||
--
|
||||
|
Reference in New Issue
Block a user