Используя этот пример данных:
CREATE TABLE MyTable (ID INT, Date DATETIME, Allocation INT);
INSERT INTO MyTable VALUES (1, {d '2012-01-01'}, 0);
INSERT INTO MyTable VALUES (2, {d '2012-01-02'}, 2);
INSERT INTO MyTable VALUES (3, {d '2012-01-03'}, 0);
INSERT INTO MyTable VALUES (4, {d '2012-01-04'}, 0);
INSERT INTO MyTable VALUES (5, {d '2012-01-05'}, 0);
INSERT INTO MyTable VALUES (6, {d '2012-01-06'}, 5);
GO
Попробуйте это:
WITH DateGroups (ID, Date, Allocation, SeedID) AS (
SELECT MyTable.ID, MyTable.Date, MyTable.Allocation, MyTable.ID
FROM MyTable
LEFT JOIN MyTable Prev ON Prev.Date = DATEADD(d, -1, MyTable.Date)
AND Prev.Allocation = 0
WHERE Prev.ID IS NULL
AND MyTable.Allocation = 0
UNION ALL
SELECT MyTable.ID, MyTable.Date, MyTable.Allocation, DateGroups.SeedID
FROM MyTable
JOIN DateGroups ON MyTable.Date = DATEADD(d, 1, DateGroups.Date)
WHERE MyTable.Allocation = 0
), StartDates (ID, StartDate, DayCount) AS (
SELECT SeedID, MIN(Date), COUNT(ID)
FROM DateGroups
GROUP BY SeedID
), EndDates (ID, EndDate) AS (
SELECT SeedID, MAX(Date)
FROM DateGroups
GROUP BY SeedID
)
SELECT StartDates.StartDate, EndDates.EndDate, StartDates.DayCount
FROM StartDates
JOIN EndDates ON StartDates.ID = EndDates.ID;
Первым разделом запроса является рекурсивный SELECT, который привязывается ко всем строкам, для которых назначено = 0, и чей предыдущий день либо не существует, либо имеет выделение! = 0. Это фактически возвращает идентификаторы: 1 и 3 которые являются начальными датами периодов времени, которые вы хотите вернуть.
Рекурсивная часть этого же запроса начинается со строк привязки и находит все последующие даты, которые также имеют распределение = 0. SeedID отслеживает привязанный идентификатор на протяжении всех итераций.
Результат пока таков:
ID Date Allocation SeedID
----------- ----------------------- ----------- -----------
1 2012-01-01 00:00:00.000 0 1
3 2012-01-03 00:00:00.000 0 3
4 2012-01-04 00:00:00.000 0 3
5 2012-01-05 00:00:00.000 0 3
Следующий подзапрос использует простой GROUP BY, чтобы отфильтровать все даты начала для каждого SeedID, а также считает дни.
Последний подзапрос делает то же самое с датами окончания, но на этот раз подсчет дней не нужен, поскольку у нас уже есть это.
Последний запрос SELECT объединяет эти два элемента, чтобы объединить даты начала и окончания, и возвращает их вместе с количеством дней.