Sto cercando un modo efficace per trovare tutte le intersezioni tra i gruppi di intervalli di data e ora. Deve funzionare con PostgreSQL 9.2.Trova tutte le intersezioni di tutti i gruppi di intervalli in PostgreSQL
Diciamo che le gamme rappresentano le volte in cui una persona è disponibile a incontrarsi. Ogni persona può avere uno o più intervalli di volte quando sono disponibili. Voglio trovare tutti i periodi di tempo in cui una riunione può avere luogo (cioè durante il quale tutte le persone sono disponibili).
Questo è quello che ho ottenuto finora. Sembra funzionare, ma non penso che sia molto efficiente, dal momento che considera la disponibilità di una persona alla volta.
WITH RECURSIVE td AS
(
-- Test data. Returns:
-- ["2014-01-20 00:00:00","2014-01-31 00:00:00")
-- ["2014-02-01 00:00:00","2014-02-20 00:00:00")
-- ["2014-04-15 00:00:00","2014-04-20 00:00:00")
SELECT 1 AS entity_id, '2014-01-01'::timestamp AS begin_time, '2014-01-31'::timestamp AS end_time
UNION SELECT 1, '2014-02-01', '2014-02-28'
UNION SELECT 1, '2014-04-01', '2014-04-30'
UNION SELECT 2, '2014-01-15', '2014-02-20'
UNION SELECT 2, '2014-04-15', '2014-05-05'
UNION SELECT 3, '2014-01-20', '2014-04-20'
)
, ranges AS
(
-- Convert to tsrange type
SELECT entity_id, tsrange(begin_time, end_time) AS the_range
FROM td
)
, min_max AS
(
SELECT MIN(entity_id), MAX(entity_id)
FROM td
)
, inter AS
(
-- Ranges for the lowest ID
SELECT entity_id AS last_id, the_range
FROM ranges r
WHERE r.entity_id = (SELECT min FROM min_max)
UNION ALL
-- Iteratively intersect with ranges for the next higher ID
SELECT entity_id, r.the_range * i.the_range
FROM ranges r
JOIN inter i ON r.the_range && i.the_range
WHERE r.entity_id > i.last_id
AND NOT EXISTS
(
SELECT *
FROM ranges r2
WHERE r2.entity_id < r.entity_id AND r2.entity_id > i.last_id
)
)
-- Take the final set of intersections
SELECT *
FROM inter
WHERE last_id = (SELECT max FROM min_max)
ORDER BY the_range;
non collegato: la fornitura di "dati statici" può essere fatto più brevi senza l'utilizzo di '' select' e union' utilizzando un 'valori 'clausola:' values (1, '2014-01-01' :: timestamp, '2014-01-31' :: timestamp), (2, ...) 'e definiscono i nomi delle colonne nel CTE. –