У меня есть таблица (2M + записи), которая отслеживает регистр.Некоторые записи добавляют очки, в то время как другие вычитают очки (есть только два вида записей).Записи, которые вычитают точки, всегда ссылаются на (добавление) записей, из которых они были вычтены с referenceentryid
.Добавляемые записи всегда будут иметь NULL
в referenceentryid
.
В этой таблице есть столбец dead
, который будет установлен на true
рабочим, когда некоторые добавления были истощены или истекли, или когдавычитание указывает на «мертвые» дополнения.Поскольку таблица имеет частичный индекс на dead=false
, SELECT в живых строках работает довольно быстро.
Моя проблема заключается в производительности работника, который устанавливает dead
в NULL
.
Поток будет: 1. Получить запись для каждого добавления, которая указывает добавленную, вычтенную сумму и истек ли срок ее действия.2. Отфильтруйте записи, которые не просрочены и имеют больше сложений, чем вычитаний.3. Обновите dead=true
в каждой строке, где либо id
, либо referenceentryid
входит в отфильтрованный набор записей.
WITH entries AS
(
SELECT
additions.id AS id,
SUM(subtractions.amount) AS subtraction,
additions.amount AS addition,
additions.expirydate <= now() AS expired
FROM
loyalty_ledger AS subtractions
INNER JOIN
loyalty_ledger AS additions
ON
additions.id = subtractions.referenceentryid
WHERE
subtractions.dead = FALSE
AND subtractions.referenceentryid IS NOT NULL
GROUP BY
subtractions.referenceentryid, additions.id
), dead_entries AS (
SELECT
id
FROM
entries
WHERE
subtraction >= addition OR expired = TRUE
)
-- THE SLOW BIT:
SELECT
*
FROM
loyalty_ledger AS ledger
WHERE
ledger.dead = FALSE AND
(ledger.id IN (SELECT id FROM dead_entries) OR ledger.referenceentryid IN (SELECT id FROM dead_entries));
В приведенном выше запросе внутренняя часть выполняется довольно быстро (несколькосекунд) пока последняя часть будет выполняться вечно.
У меня есть следующие индексы в таблице:
CREATE TABLE IF NOT EXISTS loyalty_ledger (
id SERIAL PRIMARY KEY,
programid bigint NOT NULL,
FOREIGN KEY (programid) REFERENCES loyalty_programs(id) ON DELETE CASCADE,
referenceentryid bigint,
FOREIGN KEY (referenceentryid) REFERENCES loyalty_ledger(id) ON DELETE CASCADE,
customerprofileid bigint NOT NULL,
FOREIGN KEY (customerprofileid) REFERENCES customer_profiles(id) ON DELETE CASCADE,
amount int NOT NULL,
expirydate TIMESTAMPTZ,
dead boolean DEFAULT false,
expired boolean DEFAULT false
);
CREATE index loyalty_ledger_referenceentryid_idx ON loyalty_ledger (referenceprofileid) WHERE dead = false;
CREATE index loyalty_ledger_customer_program_idx ON loyalty_ledger (customerprofileid, programid) WHERE dead = false;
Я пытаюсь оптимизировать последнюю часть запроса.EXPLAIN
дает мне следующее:
"Index Scan using loyalty_ledger_referenceentryid_idx on loyalty_ledger ledger (cost=103412.24..4976040812.22 rows=986583 width=67)"
" Filter: ((SubPlan 3) OR (SubPlan 4))"
" CTE entries"
" -> GroupAggregate (cost=1.47..97737.83 rows=252177 width=25)"
" Group Key: subtractions.referenceentryid, additions.id"
" -> Merge Join (cost=1.47..91390.72 rows=341928 width=28)"
" Merge Cond: (subtractions.referenceentryid = additions.id)"
" -> Index Scan using loyalty_ledger_referenceentryid_idx on loyalty_ledger subtractions (cost=0.43..22392.56 rows=341928 width=12)"
" Index Cond: (referenceentryid IS NOT NULL)"
" -> Index Scan using loyalty_ledger_pkey on loyalty_ledger additions (cost=0.43..80251.72 rows=1683086 width=16)"
" CTE dead_entries"
" -> CTE Scan on entries (cost=0.00..5673.98 rows=168118 width=4)"
" Filter: ((subtraction >= addition) OR expired)"
" SubPlan 3"
" -> CTE Scan on dead_entries (cost=0.00..3362.36 rows=168118 width=4)"
" SubPlan 4"
" -> CTE Scan on dead_entries dead_entries_1 (cost=0.00..3362.36 rows=168118 width=4)"
Похоже, что последняя часть моего запроса очень неэффективна.Есть идеи, как это ускорить?