2015-06-22 102 views
0

表包含特定參數的每日快照,但數據可能會在幾天內丟失。任務是計算量每月,這個緣故,我們需要在每月的開始/結束的值,如果數據丟失,我們需要對最近的日期,即:查找月份開始和結束的最近日期

[Time]     Value 
2015-04-28 00:00:00.000 76127 
2015-05-03 00:00:00.000 76879 
2015-05-22 00:00:00.000 79314 
2015-06-07 00:00:00.000 81443 

目前我使用下面的代碼:

select 
    * 
from(
    select 
    [Time], 
    Value, 
    ROW_NUMBER() over (partition by CASE WHEN [Time] < '2015-05-01' THEN 1 ELSE 0 END order by abs(DATEDIFF(DAY, '2015-05-01', [Time]))) as rn2, 
    ROW_NUMBER() over (partition by CASE WHEN [Time] > '2015-05-01' THEN 1 ELSE 0 END order by abs(DATEDIFF(DAY, [Time], '2015-05-01'))) as rn3, 
    ROW_NUMBER() over (partition by CASE WHEN [Time] < '2015-05-31' THEN 1 ELSE 0 END order by abs(DATEDIFF(DAY, '2015-05-31', [Time]))) as rn4, 
    ROW_NUMBER() over (partition by CASE WHEN [Time] > '2015-05-31' THEN 1 ELSE 0 END order by abs(DATEDIFF(DAY, [Time], '2015-05-31'))) as rn5, 
    DATEDIFF(DAY, '2015-05-01', [Time]) as doff, 
    DATEDIFF(DAY, '2015-05-31', [Time]) as doff2 
    from 
    ValueTable 
    where 
    [Time] between '2015-04-01' and '2015-06-30' 
) r 
where 
    doff = 0 or doff2 = 0 or (doff != 0 and rn2 = 1 and rn3 = 1) or (doff2 != 0 and rn4 = 1 and rn5 = 1) 

有沒有更有效的方法來做到這一點?

+0

但你的查詢不每月給量!?期望的輸出應該如何? – CeOnSql

+0

稍後將在SQL之外進行計算。 – TheeStupidOne

回答

0

下面的代碼看起來更復雜,因爲它更長。但是,它應該非常快,因爲它可以很好地利用ValueTable([Time])上的索引。

這個想法是尋找完全匹配。如果沒有完全匹配,則在日期前後查找第一個和最後一個記錄。這需要union all六個子查詢,但每個人都應該做一個索引的最佳使用:

with exact_first as (
     select t.* 
     from ValueTable t 
     where [Time] = '2015-05-01' 
    ), 
    exact_last as (
     select t.* 
     from ValueTable t 
     where [Time] = '2015-05-01' 
    ) 
(select ef.* 
from exact_first ef 
) union all 
(select top 1 t.* 
from ValueTable t 
where [Time] < '2015-05-01' and 
     not exists (select 1 from exact_first ef2) 
order by [Time] 
) union all 
(select top 1 t.* 
from ValueTable t 
where [Time] > '2015-05-01' and 
     not exists (select 1 from exact_first ef2) 
order by [Time] desc 
) union all 
(select el.* 
from exact_last el 
) union all 
(select top 1 t.* 
from ValueTable t 
where [Time] < '2015-05-31' and 
     not exists (select 1 from exact_last ef2) 
order by [Time] 
) union all 
(select top 1 t.* 
from ValueTable t 
where [Time] > '2015-05-31' and 
     not exists (select 1 from exact_last ef2) 
order by [Time] desc; 
) 
+0

謝謝戈登,它像一個魅力! – TheeStupidOne

相關問題