如何从sql中的特定日期获取四分之一的日期列表?

时间:2018-05-22 08:35:36

标签: sql-server

我想查看所提供日期的季度中的所有日期(在这种特殊情况下,我试图搜索从第一次销售的季度开始的所有销售)。

我下面的初步尝试是从提供的日期中提取月份和年份,并在查找记录时将其用作条件,遗憾的是我不知道如何提供具有多个条件的where子句。

DECLARE @FirstSale DateTime = (SELECT TOP 1 [Sale].[DateCreated] FROM [Sale] ORDER BY [Sale].[DateCreated] ASC)
DECLARE @FirstSaleMonth int = Month(@FirstSale)
DECLARE @FirstSaleYear int = Year(@FirstSale)

SELECT [Sale].[DateCreated] FROM [Sale]
WHERE Month([Sale].[DateCreated]) =  
CASE 
    WHEN @FirstSaleMonth = 1 OR @FirstSaleMonth = 2 OR @FirstSaleMonth = 3
        THEN 1 OR 2 OR 3    
    WHEN @FirstSaleMonth = 4 OR @FirstSaleMonth = 5 OR @FirstSaleMonth = 6
        Then 4 OR 5 OR 6
    WHEN @FirstSaleMonth = 7 OR @FirstSaleMonth = 8 OR @FirstSaleMonth = 9
        Then 7 OR 8 OR 9
    WHEN @FirstSaleMonth = 10 OR @FirstSaleMonth = 11 OR @FirstSaleMonth = 12
        Then 10 OR 11 OR 12
END
AND Year([Sale].[DateCreated]) = @FirstSaleYear

所以我最终做的只是重复这个条件,结果会有不同的结果,最终会达到预期的条件

WHERE Month([Sale].[DateCreated]) =  
CASE 
    WHEN @FirstSaleMonth = 1 OR @FirstSaleMonth = 2 OR @FirstSaleMonth = 3
        THEN 1
    WHEN @FirstSaleMonth = 4 OR @FirstSaleMonth = 5 OR @FirstSaleMonth = 6
        Then 4
    WHEN @FirstSaleMonth = 7 OR @FirstSaleMonth = 8 OR @FirstSaleMonth = 9
        Then 7
    WHEN @FirstSaleMonth = 10 OR @FirstSaleMonth = 11 OR @FirstSaleMonth = 12
        Then 10
END
AND Year([Sale].[DateCreated]) = @FirstSaleYear

OR

Month([Sale].[DateCreated]) = 
CASE 
    WHEN @FirstSaleMonth = 1 OR @FirstSaleMonth = 2 OR @FirstSaleMonth = 3
        THEN 2
    WHEN @FirstSaleMonth = 4 OR @FirstSaleMonth = 5 OR @FirstSaleMonth = 6
        Then 5
    WHEN @FirstSaleMonth = 7 OR @FirstSaleMonth = 8 OR @FirstSaleMonth = 9
        Then 8
    WHEN @FirstSaleMonth = 10 OR @FirstSaleMonth = 11 OR @FirstSaleMonth = 12
        Then 11
END
AND Year([Sale].[DateCreated]) = @FirstSaleYear

OR

Month([Sale].[DateCreated]) = 
CASE 
    WHEN @FirstSaleMonth = 1 OR @FirstSaleMonth = 2 OR @FirstSaleMonth = 3
        THEN 3
    WHEN @FirstSaleMonth = 4 OR @FirstSaleMonth = 5 OR @FirstSaleMonth = 6
        Then 6
    WHEN @FirstSaleMonth = 7 OR @FirstSaleMonth = 8 OR @FirstSaleMonth = 9
        Then 9
    WHEN @FirstSaleMonth = 10 OR @FirstSaleMonth = 11 OR @FirstSaleMonth = 12
        Then 12
END
AND Year([Sale].[DateCreated]) = @FirstSaleYear

但是我感觉它不必要地长,而且我已经多次使用相同的条件。我还在学习,我想知道是否有更有效的方法,以便我将来可以应用它?

1 个答案:

答案 0 :(得分:0)

那么你应该与宿舍合作。

DECLARE @FirstSale DateTime = (SELECT TOP 1 [Sale].[DateCreated] FROM [Sale] ORDER BY [Sale].[DateCreated] ASC)
DECLARE @FirstSaleQuarter int = DATEPART(QUARTER, @FirstSale)
DECLARE @FirstSaleYear int = Year(@FirstSale)

SELECT 
    [Sale].[DateCreated] 
FROM 
    [Sale]
WHERE 
    DATEPART(QUARTER, [Sale].[DateCreated]) = @FirstSaleQuarter AND 
    Year([Sale].[DateCreated]) = @FirstSaleYear

但是SARGable的最佳选择(可以正确使用索引)将使用范围(避免将函数应用于WHERE中的表格列):

DECLARE @FirstSale DateTime = (SELECT TOP 1 [Sale].[DateCreated] FROM [Sale] ORDER BY [Sale].[DateCreated] ASC)

DECLARE @StartDate DATE = DATEFROMPARTS(
    Year(@FirstSale),
    (3 * (DATEPART(QUARTER, @FirstSale) - 1) + 1,
    1)

DECLARE @EndDate DATE = DATEADD(MONTH, 3, @StartDate)

SELECT 
    [Sale].[DateCreated] 
FROM 
    [Sale]
WHERE 
    [Sale].[DateCreated] >= @StartDate AND
    [Sale].[DateCreated] < @EndDate