2011-12-06 177 views
1

我有表中的SQL Server DB:從兩列中查找最小值和最大值。生成從最小日期到最大

ID Open_date  Close_date 
1  01-01-2010 01-03-2010 
2  21-01-2011 12-02-2011 
3  01-03-2010 NULL 
4  10-01-2010 NULL 

我需要做一些T-SQL查詢將返回:條目

Month Year Open Close 
01  2010 2  0 
02  2010 0  0 
03  2010 0  1 
04  2010 0  0 
05  2010 0  0 
06  2010 0  0 
07  2010 0  0 
08  2010 0  0 
09  2010 0  0 
10  2010 0  0 
11  2010 0  0 
12  2010 0  0 
01  2011 1  0 
02  2011 0  1 

金額在結果集的equals來自DB的表的「Open_date」和「Close_date」列的最小值與來自相同列的最大值之間的月數。問題是如何從兩個日期列中查找最小值和最大值,然後生成從最小值到最大值的日期,然後使用此臨時表(或將如何)來計算臨時表中每個日期的打開和關閉條目的數量。

+0

爲什麼你要計算一個Id 2010年1月爲關閉? – Lamak

+0

對不起,我的錯。更正它。 – sunprophit

+0

我相信2010年3月線應該是1,1而不是0,1來反映第3條記錄? – Tarwn

回答

0

我改變了一些日期,因爲我使用了不同的日期系統,但這應該起作用。請接受,如果它,請!

DECLARE @TABLE1 TABLE 
(
    ID INT 
    , Open_date DATETIME 
    , Close_date DATETIME 
) 

INSERT INTO @TABLE1 (ID, Open_date, Close_date) 
     SELECT 1, '01-01-2010', '03-01-2010' 
UNION SELECT 2, '01-21-2011', '02-12-2011' 
UNION SELECT 3, '03-01-2010', NULL 
UNION SELECT 4, '01-10-2010', NULL 


DECLARE @MIN_DATE DATETIME 
DECLARE @MAX_DATE DATETIME 

SELECT @MIN_DATE = MIN(d) 
    , @MAX_DATE = MAX(d) 
FROM 
(
    SELECT Open_date AS d 
    FROM @TABLE1 

     UNION 

    SELECT Close_date AS d 
    FROM @TABLE1 
)a 

--SELECT @MIN_DATE, @MAX_DATE 



DECLARE @DATES TABLE 
(
    ID INT IDENTITY(1,1) 
    , [date] DATETIME 
) 

DECLARE @DATE DATETIME 
SET @DATE = @MIN_DATE 

WHILE (@DATE <= @MAX_DATE) 
BEGIN 

    INSERT INTO @DATES ([date]) 
    VALUES (@DATE) 

    SET @DATE = @DATE + 1 

END 

--SELECT * 
--FROM @DATES 

SELECT MONTH(a.[date]) AS [Month] 
    , YEAR(a.[date]) AS [Year] 
    , SUM(CASE WHEN b.Open_date IS NOT NULL THEN 1 ELSE 0 END) AS [Open] 
    , SUM(CASE WHEN b.Close_date IS NOT NULL THEN 1 ELSE 0 END) AS [Close] 
FROM @DATES a 
LEFT JOIN @TABLE1 b 
    ON a.[date] = b.Open_date 
GROUP BY YEAR(a.[date]), MONTH(a.[date]) 
ORDER BY YEAR(a.[date]), MONTH(a.[date]) 
1

我以前能用了一個例子,最終沒有發佈,但回來後,決定無論如何將它張貼:)

-- Sample Data 
CREATE TABLE #SampleVals (ID int, Open_Date Date, Close_Date Date); 
INSERT INTO #SampleVals(ID, Open_Date, Close_Date) 
VALUES(1,'20100101','20100301'), 
     (2,'20110121','20110212'), 
     (3,'20100301', NULL), 
     (4,'20100110',NULL); 

-- Get Start/End for full date range 
DECLARE @Min Date, @Max Date; 
SELECT @Min = DateAdd(dd,-1 * Day(MIN(Open_Date)) + 1, MIN(Open_Date)), 
     @Max = MAX(Close_Date) 
FROM #SampleVals; 

-- Query for values across entire range 
WITH DateRange (StartDate,NextDate) AS (
    SELECT DATEADD(MONTH, n-1, @Min), 
      DATEADD(MONTH, n, @Min) 
    FROM dbo.Number N 
    WHERE N.n <= DATEDIFF(MONTH,@Min,@Max) + 1 
) 
SELECT MONTH(DR.StartDate), 
     YEAR(DR.StartDate), 
     SUM(CASE WHEN S.Open_Date >= DR.StartDate Then 1 Else 0 END) AS [Open], 
     SUM(CASE WHEN S.Close_Date < DR.NextDate Then 1 Else 0 END) AS [Closed] 
FROM DateRange DR 
    LEFT JOIN #SampleVals S ON S.Open_Date < DR.NextDate 
          AND (S.Close_Date >= DR.StartDate OR S.Close_Date IS NULL) 
GROUP BY DR.StartDate 
ORDER BY DR.StartDate; 

-- Cleanup sample data 
DROP TABLE #SampleVals; 

的樣本數據的日期已更改,以反映年月日。我還用的本地號碼錶:

CREATE TABLE dbo.Number(n INT NOT NULL IDENTITY) ; 
GO 
SET NOCOUNT ON ; 
INSERT dbo.Number DEFAULT VALUES ; 
WHILE SCOPE_IDENTITY() < 5000 
    INSERT dbo.Number DEFAULT VALUES ; 

我幾乎沒有張貼此,作爲Norla得到它做得比較早之前,我完成了這一個,但我注意到Norla的解決方案填充關閉列,如果開始日期被關閉(以及開始日期的月份),而此版本在close_date的月份中填充close月份列,我相信這是您要求的。

0

對於SQL Server 2005 +,你可以嘗試

somewhat simpler approach. Try this: 

WITH CTE AS 
(
    SELECT MIN(CONVERT(VARCHAR(6),Dates,112)) YearMonth, 
      MAX(CONVERT(VARCHAR(6),Dates,112)) MaxDate 
    FROM ( SELECT Open_Date Dates FROM YourTable 
      UNION 
      SELECT Close_Date FROM YourTable) A 
    UNION ALL 
    SELECT CONVERT(VARCHAR(6),DATEADD(MONTH,1,YearMonth+'01'),112), MaxDate 
    FROM CTE 
    WHERE CONVERT(VARCHAR(6),DATEADD(MONTH,1,YearMonth+'01'),112)<=MaxDate 
) 
SELECT RIGHT(A.YearMonth,2) [Month], LEFT(A.YearMonth,4) [Year], 
     COUNT(B.Id) [Open], COUNT(C.Id) [Close] 
FROM CTE A 
LEFT JOIN YourTable B 
ON A.YearMonth = CONVERT(VARCHAR(6),B.Open_Date,112) 
LEFT JOIN YourTable C 
ON A.YearMonth = CONVERT(VARCHAR(6),C.Close_Date,112) 
GROUP BY RIGHT(A.YearMonth,2), LEFT(A.YearMonth,4) 
ORDER BY LEFT(A.YearMonth,4), RIGHT(A.YearMonth,2) 
0

另一個版本

;with T(monthstart, monthend, _tempmax) as (
    select 
     min(Open_date), dateadd(day, -1, dateadd(month, 1, min(Open_date))), dateadd(month, 1, max(Open_date)) as _tempmax from THETABLE 
    union all 
    select 
     dateadd(month, 1, monthstart), dateadd(day, -1, dateadd(month, 2, monthstart)), _tempmax 
    from T 
    where dateadd(month, 1, monthstart) <= _tempmax 
) 
select 
    month(monthstart) as [month], 
    year(monthstart) as [year], 
    sum(case when Open_date between monthstart and monthend then 1 else 0 end), 
    sum(case when Close_date between monthstart and monthend then 1 else 0 end) 
from T 
left join 
    THETABLE on (Open_date between monthstart and monthend) or (Close_date between monthstart and monthend) 
group by 
    month(monthstart), 
    year(monthstart)