SQL Server query - Fill Missing Dates In a Date-Sequenced in SQL using Tally Table

后端 未结 2 476
[愿得一人]
[愿得一人] 2021-01-25 09:50

I have a table in database with numbers of tenants, each tenant lists a record of their sales per date. There are instance where in a tenant has NO SALES in particular date/s, t

2条回答
  •  后悔当初
    2021-01-25 10:02

    You could do this using a Tally Table.

    Basically, you use the Tally Table to generate sequence of dates from @startDate to @endDate and CROSS JOIN it to DISTINCT Item to generate all Date-Item combination. Then, the result will be LEFT-JOINed to tblSales to achieve the desired output.

    SQL Fiddle

    DECLARE
        @startDate  DATE = '20140101',
        @endDate    DATE = '20140105';
    
    WITH E1(N) AS(
        SELECT 1 UNION ALL SELECT 1 UNION ALL SELECT 1 UNION ALL SELECT 1 UNION ALL SELECT 1 UNION ALL
        SELECT 1 UNION ALL SELECT 1 UNION ALL SELECT 1 UNION ALL SELECT 1 UNION ALL SELECT 1
    )
    ,E2(N) AS(SELECT 1 FROM E1 a, E1 b)
    ,E4(N) AS(SELECT 1 FROM E2 a, E2 b)
    ,Tally(N) AS(
        SELECT TOP (DATEDIFF(DAY, @startDate, @endDate) + 1) 
            ROW_NUMBER() OVER(ORDER BY(SELECT NULL))
        FROM E4
    )
    ,CteAllDates(Item, dt) AS(
        SELECT x.Item, DATEADD(DAY, N - 1, @startDate)
        FROM Tally
        CROSS JOIN(
            SELECT DISTINCT Item 
            FROM tblSales
            WHERE [Date] BETWEEN @startDate AND @endDate
        ) AS x
    )
    SELECT d.*, ts.Sales
    FROM CteAllDates d
    LEFT JOIN tblSales ts
        ON ts.Item = d.Item
        AND ts.Date = d.dt
    WHERE
        ts.[Date] BETWEEN @startDate AND @endDate
    ORDER BY d.Item, d.dt
    

    Here is an alternative. Instead of the cascading CTEs, use sys.columns to generate the Tally Table.:

    DECLARE
        @startDate  DATE = '20140101',
        @endDate    DATE = '20140105';
    
    WITH Tally(N) AS(
        SELECT TOP (DATEDIFF(DAY, @startDate, @endDate) + 1) 
            ROW_NUMBER() OVER(ORDER BY(SELECT NULL))
        FROM sys.columns a, sys.columns b
    )
    ,CteAllDates(Item, dt) AS(
        SELECT x.Item, DATEADD(DAY, N - 1, @startDate)
        FROM Tally
        CROSS JOIN(
            SELECT DISTINCT Item 
            FROM tblSales
            WHERE [Date] BETWEEN @startDate AND @endDate
        ) AS x
    )
    SELECT d.*, ts.Sales
    FROM CteAllDates d
    LEFT JOIN tblSales ts
        ON ts.Item = d.Item
        AND ts.Date = d.dt
    WHERE
        ts.[Date] BETWEEN @startDate AND @endDate
    ORDER BY d.Item, d.dt
    

    Result

    |    Item |         dt |  Sales |
    |---------|------------|--------|
    | tenant1 | 2014-01-01 |    100 |
    | tenant1 | 2014-01-02 |    100 |
    | tenant1 | 2014-01-03 |    100 |
    | tenant1 | 2014-01-04 |   NULL |
    | tenant1 | 2014-01-05 |    100 |
    | tenant2 | 2014-01-01 |    100 |
    | tenant2 | 2014-01-02 |   NULL |
    | tenant2 | 2014-01-03 |   NULL |
    | tenant2 | 2014-01-04 |    100 |
    | tenant2 | 2014-01-05 |   NULL |
    | tenant3 | 2014-01-01 |    100 |
    | tenant3 | 2014-01-02 |   NULL |
    | tenant3 | 2014-01-03 |    100 |
    | tenant3 | 2014-01-04 |   NULL |
    | tenant3 | 2014-01-05 |    100 |
    

提交回复
热议问题