12

如果我有一张看起来像这样的桌子

begin date      end date        data
 2013-01-01     2013-01-04       7
 2013-01-05     2013-01-06       9

我怎样才能让它像这样被退回......

    date         data
 2013-01-01       7
 2013-01-02       7
 2013-01-03       7
 2013-01-04       7
 2013-01-05       9
 2013-01-06       9

我正在考虑做的一件事是拥有另一个只有所有日期的表,然后使用只有日期的表加入上表date>=begin datedate<=end date但这似乎有点笨拙,不得不维护那个只有重复日期的额外表.

在某些情况下,我没有数据范围,只有一个as of日期,基本上看起来像我的第一个示例,但没有end date. end date下一行的“截至”日期暗示了这一点(即结束日期应该是下一行的-1 as of)。我对此有一个“解决方案”,它使用 row_number() 函数来获取下一个值,但我怀疑这种方法,我这样做的方式有一堆嵌套的自连接,会导致查询时间很长。

4

2 回答 2

16

使用一些样本数据...

create table data (begindate datetime, enddate datetime, data int);
insert data select 
 '20130101', '20130104', 7 union all select
 '20130105', '20130106', 9;

查询:(注意:如果您已经有一个数字/计数表 - 使用它)

select dateadd(d,v.number,d.begindate) adate, data
  from data d
  join master..spt_values v on v.type='P'
       and v.number between 0 and datediff(d, begindate, enddate)
order by adate;

结果

|                       COLUMN_0 | DATA |
-----------------------------------------
| January, 01 2013 00:00:00+0000 |    7 |
| January, 02 2013 00:00:00+0000 |    7 |
| January, 03 2013 00:00:00+0000 |    7 |
| January, 04 2013 00:00:00+0000 |    7 |
| January, 05 2013 00:00:00+0000 |    9 |
| January, 06 2013 00:00:00+0000 |    9 |

或者,您可以即时生成数字表 (0-99) 或任意数量的数字

;WITH Numbers(number) AS (
  select top(100) row_number() over (order by (select 0))-1
  from sys.columns a
  cross join sys.columns b
  cross join sys.columns c
  cross join sys.columns d
  )
select dateadd(d,v.number,d.begindate) adate, data
  from data d
  join Numbers v on v.number between 0 and datediff(d, begindate, enddate)
order by adate;

SQL 小提琴演示

于 2013-05-03T12:32:29.670 回答
2

您可以使用递归 CTE 来获取两个日期之间的所有日期。另一个 CTE 是获取 ROW_NUMBERs 来帮助您解决缺少的 EndDates。

DECLARE @startDate DATE
DECLARE @endDate DATE

SELECT @startDate = MIN(begindate) FROM Table1
SELECT @endDate = MAX(enddate) FROM Table1

;WITH CTE_Dates AS 
(
    SELECT @startDate AS DT
    UNION ALL
    SELECT DATEADD(DD, 1, DT)
    FROM CTE_Dates
    WHERE DATEADD(DD, 1, DT) <= @endDate
)
,CTE_Data AS 
(
    SELECT *, ROW_NUMBER() OVER (ORDER BY BeginDate) AS RN FROM Table1

)
SELECT DT, t1.data FROM CTE_Dates d
LEFT JOIN CTE_Data t1 on d.DT 
BETWEEN t1.[BeginDate] AND COALESCE(t1.EndDate, 
        (SELECT DATEADD(DD,-1,t2.BeginDate) FROM CTE_Data t2 WHERE t1.RN + 1 = t2.RN))

SQLFiddle 演示

于 2013-05-03T13:01:15.543 回答