Найдите промежутки во времени, не покрытые записями с датами начала и окончания - PullRequest
5 голосов
/ 03 октября 2011

У меня есть таблица записей платежей (f_fee_item) следующим образом:

Fee_Item_ID    int
Fee_Basis_ID   int
Start_Date     date
End_Date       date

(неактуальные столбцы удалены)

Предположим, что записи для одного и того же Fee_Basis_ID не будут перекрываться.

Мне нужно найти Start_Date и End_Date каждого разрыва в записях сборов для каждого Fee_Basis_ID между предоставленными @Query_Start_Date и @Query_End_Date.Мне нужны эти данные для расчета начислений за все периоды, когда сборы не начислялись.

Мне также нужен запрос для возврата записи, если для данного Fee_Basis_ID вообще нет записей об оплате (Fee_Basis_ID является иностраннымключ к D_Fee_Basis.Fee_Basis_ID, если это поможет).

Например:

@Query_Start_Date = '2011-01-01'
@Query_Start_Date = '2011-09-30'

D_Fee_Basis:

F_Fee_Item
1
2
3

F_Fee_Item:

Fee_Item_ID  Fee_Basis_ID  Start_Date  End_Date
1            1             2011-01-01  2011-03-31
2            1             2011-04-01  2011-06-30
3            2             2011-01-01  2011-03-31
4            2             2011-05-01  2011-06-30

Требуемые результаты:

Fee_Basis_ID   Start_Date  End_Date
1              2011-07-01  2011-09-30
2              2011-04-01  2011-04-30
2              2011-07-01  2011-09-30
3              2011-01-01  2011-09-30

Я несколько дней пробовал разные самостоятельные соединения, пытаясь заставить его работать, но безуспешно.

Пожалуйста, помогите !!

1 Ответ

2 голосов
/ 03 октября 2011

Вот решение:

declare @Query_Start_Date date= '2011-01-01' 
declare @Query_End_Date date = '2011-09-30' 

declare @D_Fee_Basis table(F_Fee_Item int)
insert @D_Fee_Basis values(1) 
insert @D_Fee_Basis values(2) 
insert @D_Fee_Basis values(3) 

declare @F_Fee_Item table(Fee_Item_ID int, Fee_Basis_ID int,Start_Date date,End_Date date)
insert @F_Fee_Item values(1,1,'2011-01-01','2011-03-31') 
insert @F_Fee_Item values(2,1,'2011-04-01','2011-06-30') 
insert @F_Fee_Item values(3,2,'2011-01-01','2011-03-31')
insert @F_Fee_Item values(4,2,'2011-05-01','2011-06-30')

;with a as
(-- find all days between Start_Date and End_Date
select @Query_Start_Date d
union all 
select dateadd(day, 1, d)
from a
where d <  @Query_end_Date
), b as
(--find all unused days
select a.d, F_Fee_Item Fee
from a, @D_Fee_Basis Fee
where not exists(select 1 from @F_Fee_Item where a.d between Start_Date and End_Date and Fee.F_Fee_Item = Fee_Basis_ID)
),
c as
(--find all start dates
select d, Fee, rn = row_number() over (order by fee, d) from b 
where not exists (select 1 from b b2 where dateadd(day,1, b2.d) = b.d and b2.Fee= b.Fee)
),
e as
(--find all end dates
select d, Fee, rn = row_number() over (order by fee, d) from b 
where not exists (select 1 from b b2 where dateadd(day,-1, b2.d) = b.d and b2.Fee= b.Fee)
)
--join start dates with end dates
select c.Fee Fee_Basis_ID, c.d Start_Date, e.d End_Date from c join e on c.Fee = e.Fee and c.rn = e.rn
option (maxrecursion 0)

Ссылка на результат: http://data.stackexchange.com/stackoverflow/q/114193/

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...