I have a query (executed against a Postgres (9.6.10) database using psql) that returns 10 rows. The query executes 20x slower when selecting 30 columns rather than 1.
I think I know why this is happening (see EXPLAIN Output below). And I'm guessing the workaround is to select only the id and then rejoin on the data. Does this suggest a bug in the query planner? Any other workaround?
Query 1 (executes in 20 seconds)
EXPLAIN ANALYZE SELECT fundraisers.*
FROM fundraisers
INNER JOIN audit_logs ON audit_logs.fundraiser_id = fundraisers.id
LEFT OUTER JOIN accounts ON accounts.id = fundraisers.account_id
GROUP BY accounts.id, fundraisers.id
LIMIT 10
Query 2 (executes in 1 second)
Differs only in the selected columns
EXPLAIN ANALYZE SELECT fundraisers.id
FROM fundraisers
INNER JOIN audit_logs ON audit_logs.fundraiser_id = fundraisers.id
LEFT OUTER JOIN accounts ON accounts.id = fundraisers.account_id
GROUP BY accounts.id, fundraisers.id
LIMIT 10
EXPLAIN Output
One thing I notice is that in the EXPLAIN output I see that the Hash Join has a different cost due to the width of the data being joined. ie.
-> Hash Join (cost=25967.06..109216.83 rows=1359646 width=1634) (actual time=322.987..1971.464 rows=1356192 loops=1)
vs
-> Hash Join (cost=14500.06..74422.83 rows=1359646 width=8) (actual time=111.710..730.736 rows=1356192 loops=1)
More Details
database=# EXPLAIN ANALYZE SELECT fundraisers.*
database-# FROM fundraisers
database-# INNER JOIN audit_logs ON audit_logs.fundraiser_id = fundraisers.id
database-# LEFT OUTER JOIN accounts ON accounts.id = fundraisers.account_id
database-# GROUP BY accounts.id, fundraisers.id
database-# LIMIT 10;
QUERY PLAN
---------------------------------------------------------------------------------------------------------------------------------------------------------
Limit (cost=3147608.91..3147608.98 rows=10 width=1634) (actual time=20437.137..20437.190 rows=10 loops=1)
-> Group (cost=3147608.91..3157806.25 rows=1359646 width=1634) (actual time=20437.136..20437.186 rows=10 loops=1)
Group Key: accounts.id, fundraisers.id
-> Sort (cost=3147608.91..3151008.02 rows=1359646 width=1634) (actual time=20437.133..20437.165 rows=120 loops=1)
Sort Key: accounts.id, fundraisers.id
Sort Method: external merge Disk: 1976192kB
-> Hash Join (cost=25967.06..109216.83 rows=1359646 width=1634) (actual time=322.987..1971.464 rows=1356192 loops=1)
Hash Cond: (audit_logs.fundraiser_id = fundraisers.id)
-> Seq Scan on audit_logs (cost=0.00..40634.14 rows=1517914 width=4) (actual time=0.078..324.638 rows=1517915 loops=1)
-> Hash (cost=13794.41..13794.41 rows=56452 width=1634) (actual time=321.869..321.869 rows=56452 loops=1)
Buckets: 4096 Batches: 32 Memory Usage: 2786kB
-> Hash Left Join (cost=1548.76..13794.41 rows=56452 width=1634) (actual time=16.465..122.406 rows=56452 loops=1)
Hash Cond: (fundraisers.account_id = accounts.id)
-> Seq Scan on fundraisers (cost=0.00..11546.52 rows=56452 width=1630) (actual time=0.068..54.434 rows=56452 loops=1)
-> Hash (cost=965.56..965.56 rows=46656 width=4) (actual time=16.337..16.337 rows=46656 loops=1)
Buckets: 65536 Batches: 1 Memory Usage: 2153kB
-> Seq Scan on accounts (cost=0.00..965.56 rows=46656 width=4) (actual time=0.020..8.268 rows=46656 loops=1)
Planning time: 0.748 ms
Execution time: 21013.427 ms
(19 rows)
database=# EXPLAIN ANALYZE SELECT fundraisers.id
database-# FROM fundraisers
database-# INNER JOIN audit_logs ON audit_logs.fundraiser_id = fundraisers.id
database-# LEFT OUTER JOIN accounts ON accounts.id = fundraisers.account_id
database-# GROUP BY accounts.id, fundraisers.id
database-# LIMIT 10;
QUERY PLAN
------------------------------------------------------------------------------------------------------------------------------------------------------
Limit (cost=231527.41..231527.48 rows=10 width=8) (actual time=1314.884..1314.917 rows=10 loops=1)
-> Group (cost=231527.41..241724.75 rows=1359646 width=8) (actual time=1314.884..1314.914 rows=10 loops=1)
Group Key: accounts.id, fundraisers.id
-> Sort (cost=231527.41..234926.52 rows=1359646 width=8) (actual time=1314.883..1314.901 rows=120 loops=1)
Sort Key: accounts.id, fundraisers.id
Sort Method: external merge Disk: 23840kB
-> Hash Join (cost=14500.06..74422.83 rows=1359646 width=8) (actual time=111.710..730.736 rows=1356192 loops=1)
Hash Cond: (audit_logs.fundraiser_id = fundraisers.id)
-> Seq Scan on audit_logs (cost=0.00..40634.14 rows=1517914 width=4) (actual time=0.062..224.307 rows=1517915 loops=1)
-> Hash (cost=13794.41..13794.41 rows=56452 width=8) (actual time=111.566..111.566 rows=56452 loops=1)
Buckets: 65536 Batches: 1 Memory Usage: 2687kB
-> Hash Left Join (cost=1548.76..13794.41 rows=56452 width=8) (actual time=17.362..98.257 rows=56452 loops=1)
Hash Cond: (fundraisers.account_id = accounts.id)
-> Seq Scan on fundraisers (cost=0.00..11546.52 rows=56452 width=8) (actual time=0.067..54.676 rows=56452 loops=1)
-> Hash (cost=965.56..965.56 rows=46656 width=4) (actual time=16.524..16.524 rows=46656 loops=1)
Buckets: 65536 Batches: 1 Memory Usage: 2153kB
-> Seq Scan on accounts (cost=0.00..965.56 rows=46656 width=4) (actual time=0.032..7.804 rows=46656 loops=1)
Planning time: 0.469 ms
Execution time: 1323.349 ms
GROUP BY?accountsandfundraisers.