排除周六周日的功能

Posted

技术标签:

【中文标题】排除周六周日的功能【英文标题】:Function to exclude Saturdays and Sundays 【发布时间】:2018-12-16 19:40:18 【问题描述】:

我创建了一个“每日销售查询”,用于捕获从周一至周五上午 8 点运行的前一个工作日输入的所有总销售额。

问题是,如果今天是星期一,我如何捕获星期五的记录。 这样我就可以排除周末了。

因为如果是星期一,总销售额显示为 0,这实际上是有意义的,因为星期日不是工作日。请帮忙。

查看我当前的代码:

SELECT 
    CONVERT(VARCHAR, DATEADD(dd, - 1, GETDATE()), 103) AS Date, 
    'Sales Orders' AS Type, 
    COUNT(o.SalesOrderID) AS Orders, 
    SUM(d.QtyOrdered) AS Chairs, 
    ISNULL(ROUND(SUM(d.ExtendedPrice), 2), 0) AS [Total Ex GST] 
FROM 
    dbo.SalesOrder o
LEFT OUTER JOIN 
    dbo.SalesOrderDetails d ON o.SalesOrderID = d.SalesOrderID 
WHERE 
    (o.EntryDate >= CONVERT(CHAR(8), DATEADD(dd, - 1, GETDATE()), 112)) 
    AND (o.EntryDate < CONVERT(CHAR(8), GETDATE(), 112)) 
    AND (o.CustomerID <> 187);

【问题讨论】:

学习以可读格式格式化您的 SQL。 另外,在使用(n)(var)char 声明/转换时,请务必确保声明您的长度。另请注意,“三部分和四部分列引用”。已弃用。当您引用一个列时,您只需要提供表名/别名和列名。您不应包含架构和数据库名称。 为什么要将日期转换为字符进行比较? 他们不是@DaleBurrell 只是GETDATE()。我认为他们在CONVERT(date,GETDATE()) 上很有效,因为样式 112 是 ISO 格式 (yyyyMMdd)。这是一种相当奇怪的做法,并且暗示 OP 正在使用 datetime 来存储始终具有时间值 00:00:00.000 的数据。 【参考方案1】:

您可以使用datediff(dd,0,getdate()) % 7 = 0 来确定当前日期是否为星期一,而不考虑任何其他服务器设置(这是因为 SQL Server 中的零日期是 1900-01-01,恰好是星期一)。

declare @start date;
declare @finish date;

set @start = dateadd(dd, case when datediff(dd,0,getdate()) % 7 = 0 then -3 else -1 end, getdate());
set @finish = dateadd(dd,1,@start);

select
    @start, datename(weekday,@start)
  , @finish, datename(weekday,@finish)
  , datename(weekday,getdate())
;

所以在你的查询中我会使用:

declare @start date;
declare @finish date;

set @start = dateadd(dd, case when datediff(dd,0,getdate()) % 7 = 0 then -3 else -1 end, getdate());
set @finish = dateadd(dd,1,@start);

SELECT 
    CONVERT(VARCHAR, @start, 103) AS Date, 
    'Sales Orders' AS Type, 
    COUNT(o.SalesOrderID) AS Orders, 
    SUM(d.QtyOrdered) AS Chairs, 
    ISNULL(ROUND(SUM(d.ExtendedPrice), 2), 0) AS [Total Ex GST] 
FROM 
    dbo.SalesOrder o
LEFT OUTER JOIN 
    dbo.SalesOrderDetails d ON o.SalesOrderID = d.SalesOrderID 
WHERE 
     o.EntryDate >= @start
    AND o.EntryDate < @finish
    AND o.CustomerID <> 187
;

【讨论】:

【参考方案2】:

如果您使用 case 语句来确定您需要过去多少天,例如

dateadd(dd, case when datepart(weekday,getdate()) = 1 then -3 else -1 end, getdate()) -- StartDate
dateadd(dd, case when datepart(weekday,getdate()) = 1 then -2 else 0 end, getdate()) -- EndDate

所以你的代码看起来像

SELECT 
    CONVERT(VARCHAR, dateadd(dd, case when datepart(weekday,getdate()) = 1 then -3 else -1 end, getdate()), 103) AS Date, 
    'Sales Orders' AS Type, 
    COUNT(o.SalesOrderID) AS Orders, 
    SUM(d.QtyOrdered) AS Chairs, 
    ISNULL(ROUND(SUM(d.ExtendedPrice), 2), 0) AS [Total Ex GST] 
FROM 
    dbo.SalesOrder o
LEFT OUTER JOIN 
    dbo.SalesOrderDetails d ON o.SalesOrderID = d.SalesOrderID 
WHERE 
    (o.EntryDate >= CONVERT(CHAR(8), dateadd(dd, case when datepart(weekday,getdate()) = 1 then -3 else -1 end, getdate()), 112)) 
    AND (o.EntryDate < CONVERT(CHAR(8), dateadd(dd, case when datepart(weekday,getdate()) = 1 then -2 else 0 end, getdate()), 112)) 
    AND (o.CustomerID <> 187);

PS:请确认您服务器上的工作日 1 是星期一。

【讨论】:

是的,星期一是工作日 1. 我应该把这段代码放在我的 WHERE 子句中吗? @dale burrell @TrishFernandez 查看编辑 - 仍然不确定为什么要将日期转换为 char(8)? 谢谢 :) 这行得通!抱歉问了一个愚蠢的问题,请问您为什么使用 o.EntryDate?我的意思是你为什么使用“o”。 @TrishFernandez 你的原始代码有o.?这称为表别名,是编写清晰简洁查​​询的好习惯,因为它允许您以更短的形式引用表名。如果您自己加入表(自加入),则它是必要的。

以上是关于排除周六周日的功能的主要内容,如果未能解决你的问题,请参考以下文章

JAVA 取五个工作日后的日期(仅排除周六周日)

Oracle 查询排除周末和下午 6 点到晚上 9 点

负责数据库的备份,实现周一 之 周六增量备份,周日全量备份

本月周六周日LIST集合

每月第 5 个工作日的 TimeTrigger - 计划的 Web 作业 Azure

周六和周日的行总和