add batch remove batch split batch comment selection show hidden batches hide batch highlight batch
db<>fiddle
donate feedback about
By using db<>fiddle, you agree to license everything you submit by Creative Commons CC0.
Help with an interesting Postgres question: Why isn't an Index Only Scan used on a partition accessed via the parent table?.
create schema usr;
set search_path to usr;
CREATE EXTENSION IF NOT EXISTS pg_trgm;

-- Table definitions
CREATE TABLE usr (
id bigint NOT NULL GENERATED ALWAYS AS IDENTITY,
loc_id bigint,
first_name text,
last_name text
);
CREATE TABLE loc (
id bigint NOT NULL GENERATED ALWAYS AS IDENTITY,
country text,
city text
);

-- Primary keys
ALTER TABLE ONLY usr
ADD CONSTRAINT usr_pkey PRIMARY KEY (id);
ALTER TABLE ONLY loc
ADD CONSTRAINT loc_pkey PRIMARY KEY (id);

-- Foreign key
ALTER TABLE ONLY usr--.loc_id
ADD CONSTRAINT usr_loc_id_fkey FOREIGN KEY (loc_id) REFERENCES loc(id);

-- Indices
CREATE INDEX usr_loc_id_idx ON usr
USING btree (loc_id);
CREATE INDEX usr_first_name_last_name_idx ON usr
USING gist ((first_name || ' ' || last_name) gist_trgm_ops);
CREATE INDEX loc_country_city_idx ON loc
USING gist ((country || ' ' || city) gist_trgm_ops);

CREATE SCHEMA
SET
CREATE EXTENSION
CREATE TABLE
CREATE TABLE
ALTER TABLE
ALTER TABLE
ALTER TABLE
CREATE INDEX
CREATE INDEX
CREATE INDEX
setseed
SELECT 1
INSERT 0 60000
INSERT 0 60000
--this one is quick on its own
explain analyze verbose
SELECT
usr.id AS usr_id, usr.first_name, usr.last_name,
loc.id AS loc_id, loc.country, loc.city
FROM usr
LEFT JOIN loc ON usr.loc_id = loc.id
WHERE 'baker' <% (usr.first_name || ' ' || usr.last_name)
ORDER BY 'baker' <<-> (usr.first_name || ' ' || usr.last_name)
--LIMIT 20;
QUERY PLAN
Sort (cost=2793.39..2794.78 rows=556 width=148) (actual time=14.499..14.502 rows=0 loops=1)
  Output: usr.id, usr.first_name, usr.last_name, loc.id, loc.country, loc.city, (('baker'::text <<-> ((usr.first_name || ' '::text) || usr.last_name)))
  Sort Key: (('baker'::text <<-> ((usr.first_name || ' '::text) || usr.last_name)))
  Sort Method: quicksort Memory: 25kB
  -> Hash Left Join (cost=2030.92..2768.04 rows=556 width=148) (actual time=14.484..14.487 rows=0 loops=1)
        Output: usr.id, usr.first_name, usr.last_name, loc.id, loc.country, loc.city, ('baker'::text <<-> ((usr.first_name || ' '::text) || usr.last_name))
        Inner Unique: true
        Hash Cond: (usr.loc_id = loc.id)
        -> Bitmap Heap Scan on usr.usr (cost=105.98..837.46 rows=556 width=80) (actual time=14.484..14.484 rows=0 loops=1)
              Output: usr.id, usr.loc_id, usr.first_name, usr.last_name
              Filter: ('baker'::text <% ((usr.first_name || ' '::text) || usr.last_name))
              -> Bitmap Index Scan on usr_first_name_last_name_idx (cost=0.00..105.84 rows=556 width=0) (actual time=14.475..14.475 rows=0 loops=1)
                    Index Cond: (((usr.first_name || ' '::text) || usr.last_name) %> 'baker'::text)
        -> Hash (cost=1234.42..1234.42 rows=55242 width=72) (never executed)
              Output: loc.id, loc.country, loc.city
              -> Seq Scan on usr.loc (cost=0.00..1234.42 rows=55242 width=72) (never executed)
                    Output: loc.id, loc.country, loc.city
Planning Time: 0.561 ms
Execution Time: 14.624 ms
EXPLAIN
--this one is also quick on its own
explain analyze verbose
SELECT
usr.id AS usr_id, usr.first_name, usr.last_name,
loc.id AS loc_id, loc.country, loc.city
FROM usr
LEFT JOIN loc ON usr.loc_id = loc.id
WHERE 'baker' <% (loc.country || ' ' || loc.city)
ORDER BY ('baker' <<-> (loc.country || ' ' || loc.city))
--LIMIT 20;
QUERY PLAN
Sort (cost=2271.61..2273.00 rows=555 width=148) (actual time=14.286..14.289 rows=0 loops=1)
  Output: usr.id, usr.first_name, usr.last_name, loc.id, loc.country, loc.city, (('baker'::text <<-> ((loc.country || ' '::text) || loc.city)))
  Sort Key: (('baker'::text <<-> ((loc.country || ' '::text) || loc.city)))
  Sort Method: quicksort Memory: 25kB
  -> Hash Join (cost=799.51..2246.31 rows=555 width=148) (actual time=14.273..14.276 rows=0 loops=1)
        Output: usr.id, usr.first_name, usr.last_name, loc.id, loc.country, loc.city, ('baker'::text <<-> ((loc.country || ' '::text) || loc.city))
        Inner Unique: true
        Hash Cond: (usr.loc_id = loc.id)
        -> Seq Scan on usr.usr (cost=0.00..1296.75 rows=55575 width=80) (actual time=0.014..0.014 rows=1 loops=1)
              Output: usr.id, usr.loc_id, usr.first_name, usr.last_name
        -> Hash (cost=792.61..792.61 rows=552 width=72) (actual time=14.251..14.252 rows=0 loops=1)
              Output: loc.id, loc.country, loc.city
              Buckets: 1024 Batches: 1 Memory Usage: 8kB
              -> Bitmap Heap Scan on usr.loc (cost=105.94..792.61 rows=552 width=72) (actual time=14.250..14.251 rows=0 loops=1)
                    Output: loc.id, loc.country, loc.city
                    Filter: ('baker'::text <% ((loc.country || ' '::text) || loc.city))
                    -> Bitmap Index Scan on loc_country_city_idx (cost=0.00..105.80 rows=552 width=0) (actual time=14.246..14.247 rows=0 loops=1)
                          Index Cond: (((loc.country || ' '::text) || loc.city) %> 'baker'::text)
Planning Time: 0.191 ms
Execution Time: 14.334 ms
EXPLAIN
--this one's dog slow
explain analyze verbose
SELECT
usr.id AS usr_id, usr.first_name, usr.last_name,
loc.id AS loc_id, loc.country, loc.city
FROM usr
LEFT JOIN loc ON usr.loc_id = loc.id
WHERE
'baker' <% (usr.first_name || ' ' || usr.last_name)
OR
'baker' <% (loc.country || ' ' || loc.city)
ORDER BY
('baker' <<-> (usr.first_name || ' ' || usr.last_name))
+
('baker' <<-> (loc.country || ' ' || loc.city));
QUERY PLAN
Sort (cost=3442.88..3445.64 rows=1106 width=148) (actual time=877.859..877.862 rows=0 loops=1)
  Output: usr.id, usr.first_name, usr.last_name, loc.id, loc.country, loc.city, ((('baker'::text <<-> ((usr.first_name || ' '::text) || usr.last_name)) + ('baker'::text <<-> ((loc.country || ' '::text) || loc.city))))
  Sort Key: ((('baker'::text <<-> ((usr.first_name || ' '::text) || usr.last_name)) + ('baker'::text <<-> ((loc.country || ' '::text) || loc.city))))
  Sort Method: quicksort Memory: 25kB
  -> Hash Left Join (cost=1924.95..3386.96 rows=1106 width=148) (actual time=877.854..877.856 rows=0 loops=1)
        Output: usr.id, usr.first_name, usr.last_name, loc.id, loc.country, loc.city, (('baker'::text <<-> ((usr.first_name || ' '::text) || usr.last_name)) + ('baker'::text <<-> ((loc.country || ' '::text) || loc.city)))
        Inner Unique: true
        Hash Cond: (usr.loc_id = loc.id)
        Filter: (('baker'::text <% ((usr.first_name || ' '::text) || usr.last_name)) OR ('baker'::text <% ((loc.country || ' '::text) || loc.city)))
        Rows Removed by Filter: 60000
        -> Seq Scan on usr.usr (cost=0.00..1296.75 rows=55575 width=80) (actual time=0.008..9.546 rows=60000 loops=1)
              Output: usr.id, usr.loc_id, usr.first_name, usr.last_name
        -> Hash (cost=1234.42..1234.42 rows=55242 width=72) (actual time=22.177..22.178 rows=60000 loops=1)
              Output: loc.id, loc.country, loc.city
              Buckets: 65536 Batches: 1 Memory Usage: 5786kB
              -> Seq Scan on usr.loc (cost=0.00..1234.42 rows=55242 width=72) (actual time=0.009..7.300 rows=60000 loops=1)
                    Output: loc.id, loc.country, loc.city
Planning Time: 0.175 ms
Execution Time: 877.918 ms
EXPLAIN
--this one runs the two quick ones separately, deduplicates and orders them
EXPLAIN ANALYSE VERBOSE
SELECT * FROM
( SELECT
usr.id AS usr_id, usr.first_name, usr.last_name,
loc.id AS loc_id, loc.country, loc.city
FROM usr
LEFT JOIN loc ON usr.loc_id = loc.id
WHERE 'baker' <% (loc.country || ' ' || loc.city)
UNION
SELECT
usr.id AS usr_id, usr.first_name, usr.last_name,
loc.id AS loc_id, loc.country, loc.city
FROM usr
LEFT JOIN loc ON usr.loc_id = loc.id
WHERE 'baker' <% (usr.first_name || ' ' || usr.last_name)
) ORDER BY
('baker' <<-> (first_name || ' ' || last_name))
+
('baker' <<-> (country || ' ' || city));
QUERY PLAN
Sort (cost=5112.22..5115.00 rows=1111 width=148) (actual time=28.785..28.791 rows=0 loops=1)
  Output: unnamed_subquery.usr_id, unnamed_subquery.first_name, unnamed_subquery.last_name, unnamed_subquery.loc_id, unnamed_subquery.country, unnamed_subquery.city, ((('baker'::text <<-> ((unnamed_subquery.first_name || ' '::text) || unnamed_subquery.last_name)) + ('baker'::text <<-> ((unnamed_subquery.country || ' '::text) || unnamed_subquery.city))))
  Sort Key: ((('baker'::text <<-> ((unnamed_subquery.first_name || ' '::text) || unnamed_subquery.last_name)) + ('baker'::text <<-> ((unnamed_subquery.country || ' '::text) || unnamed_subquery.city))))
  Sort Method: quicksort Memory: 25kB
  -> Subquery Scan on unnamed_subquery (cost=5025.47..5056.02 rows=1111 width=148) (actual time=28.779..28.785 rows=0 loops=1)
        Output: unnamed_subquery.usr_id, unnamed_subquery.first_name, unnamed_subquery.last_name, unnamed_subquery.loc_id, unnamed_subquery.country, unnamed_subquery.city, (('baker'::text <<-> ((unnamed_subquery.first_name || ' '::text) || unnamed_subquery.last_name)) + ('baker'::text <<-> ((unnamed_subquery.country || ' '::text) || unnamed_subquery.city)))
        -> HashAggregate (cost=5025.47..5036.58 rows=1111 width=144) (actual time=28.778..28.783 rows=0 loops=1)
              Output: usr.id, usr.first_name, usr.last_name, loc.id, loc.country, loc.city
              Group Key: usr.id, usr.first_name, usr.last_name, loc.id, loc.country, loc.city
              Batches: 1 Memory Usage: 73kB
              -> Append (cost=798.13..5008.80 rows=1111 width=144) (actual time=28.758..28.763 rows=0 loops=1)
                    -> Hash Join (cost=798.13..2240.77 rows=555 width=144) (actual time=14.328..14.331 rows=0 loops=1)
                          Output: usr.id, usr.first_name, usr.last_name, loc.id, loc.country, loc.city
                          Inner Unique: true
                          Hash Cond: (usr.loc_id = loc.id)
                          -> Seq Scan on usr.usr (cost=0.00..1296.75 rows=55575 width=80) (actual time=0.007..0.007 rows=1 loops=1)
                                Output: usr.id, usr.loc_id, usr.first_name, usr.last_name
                          -> Hash (cost=791.23..791.23 rows=552 width=72) (actual time=14.316..14.318 rows=0 loops=1)
                                Output: loc.id, loc.country, loc.city
                                Buckets: 1024 Batches: 1 Memory Usage: 8kB
                                -> Bitmap Heap Scan on usr.loc (cost=104.56..791.23 rows=552 width=72) (actual time=14.315..14.316 rows=0 loops=1)
                                      Output: loc.id, loc.country, loc.city
                                      Filter: ('baker'::text <% ((loc.country || ' '::text) || loc.city))
                                      -> Bitmap Index Scan on loc_country_city_idx (cost=0.00..104.42 rows=552 width=0) (actual time=14.311..14.311 rows=0 loops=1)
                                            Index Cond: (((loc.country || ' '::text) || loc.city) %> 'baker'::text)
                    -> Hash Left Join (cost=2029.53..2762.48 rows=556 width=144) (actual time=14.422..14.424 rows=0 loops=1)
                          Output: usr_1.id, usr_1.first_name, usr_1.last_name, loc_1.id, loc_1.country, loc_1.city
                          Inner Unique: true
                          Hash Cond: (usr_1.loc_id = loc_1.id)
                          -> Bitmap Heap Scan on usr.usr usr_1 (cost=104.59..836.07 rows=556 width=80) (actual time=14.421..14.422 rows=0 loops=1)
                                Output: usr_1.id, usr_1.loc_id, usr_1.first_name, usr_1.last_name
                                Filter: ('baker'::text <% ((usr_1.first_name || ' '::text) || usr_1.last_name))
                                -> Bitmap Index Scan on usr_first_name_last_name_idx (cost=0.00..104.45 rows=556 width=0) (actual time=14.417..14.417 rows=0 loops=1)
                                      Index Cond: (((usr_1.first_name || ' '::text) || usr_1.last_name) %> 'baker'::text)
                          -> Hash (cost=1234.42..1234.42 rows=55242 width=72) (never executed)
                                Output: loc_1.id, loc_1.country, loc_1.city
                                -> Seq Scan on usr.loc loc_1 (cost=0.00..1234.42 rows=55242 width=72) (never executed)
                                      Output: loc_1.id, loc_1.country, loc_1.city
Planning Time: 0.309 ms
Execution Time: 28.912 ms
EXPLAIN