SQL SERVER - 把星期一(周一)当作每个星期的开始在一年中求取周数

浏览: 5737

开篇介绍

今天想写一个有关计算 Week Number 的函数,刚开始觉得应该很简单,凭着感觉七写八写到最后发现越写越乱,到最后搞了快两个小时以为解决了,结果一测还有好多数据不正确。非常有挫败感!感觉很不服气,觉得很丢人,跑出去站了会,冷静下来,重新拿起纸笔认真的分析了一下,连写到测试快半个小时还是解决了。

在 SQL Server 中默认情况下,每周的开始都是从周日开始算起的。但是在国内也有不一样的要求,比如按照习惯往往要求每周从周一算起。这样一来之前在数据仓库中的 Week Number 可能就不准确了,因为可能很多时候都是按默认方式来生成 Week Number 的。

解决的方式

这里有三种方式可以解决这个问题:

第一种方式是直接通过 SET DATEFIRST VALUE 来更改重新生成新的 DimDate,然后每次需要单独计算 Week Number 的时候根据 Date Key 关联一下就可以了,但这样就需要不断 JOIN DimDate,每一条记录都要 LookUp 一遍,不太好。

第二种方式就是在存储过程中需要使用到  Week Number 的时候,就先设置一下 SET DATEFIRST 然后在使用 DATEPART() 函数来获取 Week Number。这种方式需要每次都显示的  SET DATEFIRST ,不太方便。

第三种方式就是直接写一个函数,每次调用一下就可以了。

关于使用 SET DATEFIRST <VALUE> - <VALUE> 的值从 1 到 7,即周一到周日,默认值是 7。

-- The default first date in a week is Sunday, the value is 7
SELECT @@DATEFIRST

-- Default DATEFIRST is Sunday
SELECT DATENAME(WEEK,'2013-12-31') AS WeekName -- 53
SELECT DATENAME(WEEK,'2014-01-01') AS WeekName -- 1
SELECT DATENAME(WEEK,'2014-01-05') AS WeekName -- 2

-- Change the DATEFIRST to 1, Monday will be the first day of week.
SET DATEFIRST 1

SELECT @@DATEFIRST -- 1

-- After change the DATEFIRST to Monday
SELECT DATENAME(WEEK,'2013-12-31') AS WeekName -- 53
SELECT DATENAME(WEEK,'2014-01-01') AS WeekName -- 1
SELECT DATENAME(WEEK,'2014-01-05') AS WeekName -- 1

要注意的是 SET DATEFIRST 只在当前执行中有效,也就说比如新开一个查询页面继续查询 SELECT @@DATEFIRST 则还是显示默认值 7。

在创建时间维度的代码中添加 SET DATEFIRST 1,表示每周以周一开始。

USE BIWORK_SSIS
GO
SET NOCOUNT ON

-- 设置每周的起始天为周一
SET DATEFIRST 1

IF OBJECT_ID('DimDateStartWithMonday','U') IS NOT NULL
DROP TABLE DimDateStartWithMonday
GO

CREATE TABLE DimDateStartWithMonday
(
DateKey INT PRIMARY KEY,
FullDate DATE NOT NULL,
[DateName] NVARCHAR(20),
DayNumberOfWeek TINYINT NOT NULL,
DayNameOfWeek NVARCHAR(10) NOT NULL,
DayNumberOfMonth TINYINT NOT NULL,
DayNumberOfYear SMALLINT NOT NULL,
WeekNumberOfYear TINYINT NOT NULL,
EnglishMonthName NVARCHAR(10) NOT NULL,
MonthNumberOfYear TINYINT NOT NULL,
CalendarQuarter TINYINT NOT NULL,
CalendarSemester TINYINT NOT NULL,
CalendarYear SMALLINT NOT NULL
)

DECLARE @StartDate DATETIME
DECLARE @EndDate DATETIME

SELECT @StartDate = '2001-01-01',
@EndDate = '2035-12-31'

WHILE (@StartDate <= @EndDate)
BEGIN
INSERT INTO DimDateStartWithMonday
(
DateKey,
FullDate,
[DateName],
DayNumberOfWeek,
DayNameOfWeek,
DayNumberOfMonth,
DayNumberOfYear,
WeekNumberOfYear,
EnglishMonthName,
MonthNumberOfYear,
CalendarQuarter,
CalendarSemester,
CalendarYear
)
SELECT CAST(CONVERT(VARCHAR(8),@StartDate,112) AS INT) AS DateKey,
CONVERT(VARCHAR(10), @StartDate,20) AS FullDate,
CONVERT(VARCHAR(20), @StartDate,106) AS [DateName],
DATEPART(DW,@StartDate) AS DayNumberOfWeek,
DATENAME(DW,@StartDate) AS DayNameOfWeek,
DATENAME(DD,@StartDate) AS [DayOfMonth],
DATENAME(DY,@StartDate) AS [DayOfYear],
DATEPART(WW,@StartDate) AS WeekNumberOfYear,
DATENAME(MM,@StartDate) AS EnglishMonthName,
DATEPART(MM,@StartDate) AS MonthNumberOfYear,
DATEPART(QQ,@StartDate) AS CalendarQuarter,
CASE WHEN DATEPART(MM,@StartDate) BETWEEN 1 AND 6
THEN 1
ELSE 2
END AS CalendarSemester,
DATEPART(YY,@StartDate) AS CalendarYear

SET @StartDate = @StartDate + 1
END
GO

最后是函数,这个函数麻烦的地方就是要考虑周日的情况。默认情况下,周日是每个星期的第一天,但是这里改成了周一是每周的第一天,逻辑上就会复杂很多。

比如,2012-01-01 是周日,2012-01-02 是周一。按默认情况,这两天的 Week Number 都是 1,但是这里需要把 2012-01-02 的 Week Number 变成 2。

比如,2011-01-01 是周六,2012-01-02 是周日。按默认情况,周六的 Week Number 是 1, 周日的是 2。但是这里需要把周六和周日的都变成 1, 周一的变成 2。

除此之外,还要考虑之后的每一个周日与周一的交替情况。

USE BIWORK_SSIS
GO

IF OBJECT_ID('ETLWORK_GETWEEKNUMBER','FN') IS NOT NULL
DROP FUNCTION ETLWORK_GETWEEKNUMBER
GO

CREATE FUNCTION ETLWORK_GETWEEKNUMBER(@DATE DATETIME)
RETURNS INTEGER
AS
BEGIN

DECLARE @FIRST_DATE_OF_YEAR DATETIME = DATEADD(YYYY,DATEDIFF(YYYY,0,@DATE),0)
-- DECLARE @MONDAY_OF_WEEK DATETIME = DATEADD(WK,DATEDIFF(WK,0,@DATE),0)
-- DECLARE @PREVIOUS_DATE DATETIME = DATEADD(DAY,-1,@DATE)
DECLARE @WEEK_NUMBER INTEGER

-- 如果当前时间是当前年的第一天
IF @DATE = @FIRST_DATE_OF_YEAR
SET @WEEK_NUMBER = 1
-- 星期天是年第一天的情况
ELSE IF (DATEPART(WEEKDAY,@DATE) = 1 AND DATEDIFF(DAYOFYEAR,@FIRST_DATE_OF_YEAR,@DATE)/7 + 1 = DATEPART(WEEK,@DATE))
SET @WEEK_NUMBER = DATEPART(WEEK,@DATE)
-- 星期天不是年第一天的情况
ELSE IF (DATEPART(WEEKDAY,@DATE) = 1 AND DATEDIFF(DAYOFYEAR,@FIRST_DATE_OF_YEAR,@DATE)/7 + 1 <> DATEPART(WEEK,@DATE))
SET @WEEK_NUMBER = DATEPART(WEEK,@DATE) - 1
-- 如果当前天的上一个周日小于年第一天
ELSE IF DATEADD(DAY,-1,DATEADD(WK,DATEDIFF(WK,0,@DATE),0)) < @FIRST_DATE_OF_YEAR
SET @WEEK_NUMBER = 1
-- 当前天前面的一个周日正好是以周日为开始年的 7 倍的天数
ELSE IF DATEDIFF(DAYOFYEAR,@FIRST_DATE_OF_YEAR,DATEADD(DAY,-1,DATEADD(WK,DATEDIFF(WK,0,@DATE),0) ))/7 + 1 = DATEPART(WEEK,@DATE)
SET @WEEK_NUMBER = DATEPART(WEEK,@DATE) + 1
ELSE
SET @WEEK_NUMBER = DATEPART(WEEK,@DATE)

RETURN @WEEK_NUMBER
END
GO

为了方便理解,可以查看下面的查询。

DECLARE @DATE DATETIME = '2012-01-29'
DECLARE @FIRST_DATE_OF_YEAR DATETIME = DATEADD(YYYY,DATEDIFF(YYYY,0,@DATE),0)

SELECT DATEPART(WEEK,@DATE), -- 一年中的周数,默认以周日开始
DATEADD(WK,DATEDIFF(WK,0,@DATE),0), -- 当前周的周一,默认从周日开始,但是仍然找周一
DATEADD(DAY,-1,DATEADD(WK,DATEDIFF(WK,0,@DATE),0)), -- 当前周先找周一,然后往前一天找到周日
DATEDIFF(DAYOFYEAR,@FIRST_DATE_OF_YEAR,DATEADD(DAY,-1,DATEADD(WK,DATEDIFF(WK,0,@DATE),0))), -- 当前天离年第一天的间隔
DATEDIFF(DAYOFYEAR,@FIRST_DATE_OF_YEAR,DATEADD(DAY,-1,DATEADD(WK,DATEDIFF(WK,0,@DATE),0) ))/7 + 1 -- 按天计算的周数

测试一下,查找不匹配的 Week Number - 30多年的数据结果都匹配,记得要新开一个页面,以免之前 SET DATEFIRST 的影响。

查询部分数据的 WeekNumber。

当然,我感觉写的还是有点复杂,谁解决过类似问题的,期望有人能提出更简洁的写法。

推荐 0
本文由 BIWORK 创作,采用 知识共享署名-相同方式共享 3.0 中国大陆许可协议 进行许可。
转载、引用前需联系作者,并署名作者且注明文章出处。
本站文章版权归原作者及原出处所有 。内容为作者个人观点, 并不代表本站赞同其观点和对其真实性负责。本站是一个个人学习交流的平台,并不用于任何商业目的,如果有任何问题,请及时联系我们,我们将根据著作权人的要求,立即更正或者删除有关内容。本站拥有对此声明的最终解释权。

1 个评论

也思考许久,也算是模拟了sql server 中的datepart(week,@dtmDate)这个功能,将@@datefirst这个全局变量作为变量的参数来处理,可以动态指定一周的第一天的索引值(索引值从1开始计数,下同)。仅仅实现了功能而已,没有进行面向对象的单一职责的封装。大致实现思路如下:
1、根据指定日期和一周第一天索引值来获取指定日期在当前周的日索引值;
2、在获取指定日期当前年的第一天和当前年的第一周第一天;
3、根据指定日期、当前年的第一天、当前年的第一周第一天这三个日期进行逻辑处理判定:
3.1、如果指定日期大于等于当前年的第一天且指定日期小于当前年的第一周第一天时,当前星期索引数默认为1。
3.2、第一步:3.1的否定为真时,则计算指定日期与当前年的第一周第一天的日时刻粒度的间隔出于7在默认加上1;
第二步:如果当前年的第一天小于当前年的第一周第一天时,3.2第一步的结果值在加上1即可。
具体的代码实现如下:
IF OBJECT_ID(N'[dbo].[ufn_WeekOfYear]', 'FN') IS NOT NULL
BEGIN
DROP FUNCTION [dbo].[ufn_WeekOfYear];
END
GO

--==================================
-- 功能: 获取指定日期所在当前年中的星期索引值(索引值从1开始计数,大致表示为1、2、3、……、51、52、53)
-- 说明: 具体实现阐述
-- 创建: 2016-07-05 21:40-22:40 剑走江湖 实现功能
-- 修改: yyyy-MM-dd XXX 修改内容描述
--==================================
CREATE FUNCTION [dbo].[ufn_WeekOfYear]
(
@dtmDate AS DATETIME ---指定日期
,@tntDateFirst AS TINYINT = 1 --一周第一天(1、2、3、……、7分别对应周一、周二、周三、……、周日),默认为周一为一周第一天
) RETURNS TINYINT
AS
BEGIN
--一周第一天参数逻辑处理
IF (@tntDateFirst IS NULL OR @tntDateFirst = 0 OR @tntDateFirst NOT BETWEEN 1 AND 7)
BEGIN
SET @tntDateFirst = 1;
END

--声明当前年星期索引的变量,且赋值为1(表示当前年第一周)
DECLARE @tntWeekOfYear AS TINYINT;
SET @tntWeekOfYear = 1;
--声明当前年的第一天的变量且赋值
DECLARE @dtmFirstDayOfYear AS DATETIME;
SET @dtmFirstDayOfYear = DATEADD(YEAR, DATEDIFF(YEAR, 0, @dtmDate), 0);

--声明当前年的第一周第一天的变量,且赋值为当前年的第一天
DECLARE @dtmFirstWeekFirstDayOfYear AS DATETIME;
SET @dtmFirstWeekFirstDayOfYear = @dtmFirstDayOfYear;

--声明开始日期和结束日期变量且赋值主要用来获取当前周的
DECLARE
@dtmStartDate AS DATETIME
,@dtmEndDate AS DATETIME;
SELECT
@dtmStartDate = @dtmFirstDayOfYear
,@dtmEndDate = DATEADD(DAY, 7, @dtmFirstDayOfYear);

--获取当前年的第一周第一天的算法实现
WHILE (@dtmStartDate <= @dtmEndDate)
BEGIN
--获取指定日期所在当前周的日索引值(索引值从1开始计数,1、2、3、……、6、7分别对应周一、周二、周三、……、周六、周日)
--使用(@@datefirt + datepart(weekday, @dtmDate))%7的结果值从2、3、4、5、6、0、1
--分别对应周一、周二、周三、周四、周五、周六、周日
DECLARE @tntDayOfWeek AS TINYINT;
SET @tntDayOfWeek = 1;
DECLARE @tntRemainder AS TINYINT;
SET @tntRemainder = (@@DATEFIRST + DATEPART(WEEKDAY, @dtmStartDate)) % 7;
SET @tntDayOfWeek = CASE WHEN @tntRemainder <= 1 THEN @tntRemainder + 6 ELSE @tntRemainder - 1 END;

IF (@tntDayOfWeek = @tntDateFirst)
BEGIN
SET @dtmFirstWeekFirstDayOfYear = @dtmStartDate;

BREAK;
END

SET @dtmStartDate = DATEADD(DAY, 1, @dtmStartDate);
END

--获取当前年的星期索引的算法实现
IF NOT(@dtmDate >= @dtmFirstDayOfYear AND @dtmDate < @dtmFirstWeekFirstDayOfYear)
BEGIN
SET @tntWeekOfYear = @tntWeekOfYear + DATEDIFF(DAY, @dtmFirstWeekFirstDayOfYear, @dtmDate) / 7;

IF(@dtmFirstDayOfYear < @dtmFirstWeekFirstDayOfYear)
BEGIN
SET @tntWeekOfYear = @tntWeekOfYear + 1;
END
END

RETURN @tntWeekOfYear;
END
GO

测试代码你测试哈,这个实现楼主可以继续分解和封装一下函数:
1、获取指定日期当前年的第一天的函数。
2、获取指定日期当前周的日索引数函数。
3、获取指定日期当前年的第一周第一天的函数。
以上分解每个功能比较单一,也很容易维护。
测试代码就不写啦,希望能帮助到楼主。

要回复文章请先登录注册