- Перейти к 2-й последний блок кода для:
*I want to be able to tell that
between 1/10/11 and 1/13/11 there is
a gap so the start and end date is*
невозможно.
- Перейти к последнему блоку кода для:
*I want to return only
the datespans up to the first gap
encountered.*
Прежде всего, вот виртуальный стол для обсуждения
create table spans (date1 datetime, date2 datetime);
insert into spans select '20110101', '20110110';
insert into spans select '20110113', '20110115';
insert into spans select '20110120', '20110130';
Это запрос, который в отдельности перечислит все даты в календаре
declare @startdate datetime, @enddate datetime
select @startdate = '20110107', @enddate = '20110114'
select distinct a.date1+v.number
from spans A
inner join master..spt_values v
on v.type='P' and v.number between 0 and datediff(d, a.date1, a.date2)
-- we don't care about spans that don't intersect with our range
where A.date1 <= @enddate
and @startdate <= A.date2
Вооружившись этим запросом, мы теперь можем проверить, есть ли пробелы,
подсчет дней в календаре против ожидаемого количества дней
declare @startdate datetime, @enddate datetime
select @startdate = '20110107', @enddate = '20110114'
select case when count(distinct a.date1+v.number)
= datediff(d,@startdate, @enddate) + 1
then 'No gaps' else 'Gap' end
from spans A
inner join master..spt_values v
on v.type='P' and v.number between 0 and datediff(d, a.date1, a.date2)
-- we don't care about spans that don't intersect with our range
where A.date1 <= @enddate
and @startdate <= A.date2
-- count only those dates within our range
and a.date1 + v.number between @startdate and @enddate
Еще один способ сделать это - просто построить календарь из @start.
@ конец вперед и посмотреть, есть ли промежуток с этой датой
declare @startdate datetime, @enddate datetime
select @startdate = '20110107', @enddate = '20110114'
-- startdate+v.number is a day on the calendar
select @startdate + v.number
from master..spt_values v
where v.type='P' and v.number between 0
and datediff(d, @startdate, @enddate)
-- run the part above this line alone to see the calendar
-- the condition checks for dates that are not in any span (gap)
and not exists (
select *
from spans
where @startdate + v.number between date1 and date2)
Запрос возвращает ВСЕ даты, которые являются пробелами в диапазоне дат @start - @end
TOP 1
можно добавить, чтобы увидеть, есть ли пробелы
Чтобы вернуть все записи, находящиеся до пробела, используйте запрос как
производная таблица в большем запросе
declare @startdate datetime, @enddate datetime
select @startdate = '20110107', @enddate = '20110114'
select *
from spans
where date1 <= @enddate and @startdate <= date2 -- overlaps
and date2 < ( -- before the gap
select top 1 @startdate + v.number
from master..spt_values v
where v.type='P' and v.number between 0
and datediff(d, @startdate, @enddate)
and not exists (
select *
from spans
where @startdate + v.number between date1 and date2)
order by 1 ASC
)