SQL生成日历

http://jackywood.itpub.net/post/1369/127565

一条SQL语句生成年历。

  1. select case
  2.            when (new_yweek = min(new_yweek)over(partition by mon order by new_yweek)) then
  3.              mon_name
  4.            else
  5.              null
  6.          end as month,
  7.          new_yweek as yweek,
  8.          row_number() over(partition by mon order by new_yweek) as mweek,
  9.          sum(decode(wday, '1', mday, null)) as sun,
  10.          sum(decode(wday, '2', mday, null)) as mon,
  11.          sum(decode(wday, '3', mday, null)) as tue,
  12.          sum(decode(wday, '4', mday, null)) as wed,
  13.          sum(decode(wday, '5', mday, null)) as thu,
  14.          sum(decode(wday, '6', mday, null)) as fri,
  15.          sum(decode(wday, '7', mday, null)) as sat
  16.     from (select dayofyear as everyday,
  17.                  to_char(dayofyear, 'mm') as mon,
  18.                  to_char(dayofyear, 'Month') as mon_name,
  19.                  to_char(dayofyear, 'w') as mweek,
  20.                  to_char(dayofyear, 'ww') as yweek,
  21.                  case
  22.                    when (to_char(to_date(&year || '0101', 'yyyymmdd'), 'd') > '1') and
  23.                         (to_char(dayofyear, 'd') <
  24.                         to_char(to_date(&year || '0101', 'yyyymmdd'), 'd')) then
  25.                     to_char(to_char(dayofyear, 'ww') + 1, 'fm00')
  26.                    else
  27.                     to_char(dayofyear, 'ww')
  28.                  end as new_yweek,
  29.                  to_char(dayofyear, 'd') as wday,
  30.                  to_char(dayofyear, 'dd') as mday
  31.             from (select to_date(&year || '0101', 'yyyymmdd') + level - 1 as dayofyear
  32.                     from dual
  33.                   connect by level <= to_char(to_date(&year || '1231', 'yyyymmdd'),'ddd')
  34.                  )
  35.          )
  36.    group by mon, mon_name, new_yweek
  37.   /
链接是作者的解读,我感觉年历的实现主要有三个步骤
1.生成一年之中所有的日期
2.在上步的基础上,得到每个日期所在周、月、年的具体信息。
3.行列转置


其中new_yweek这部分看的真是云里雾里,它主要解决Oracle to_char函数IW和WW坑爹的问题。
select to_char(d,'yyyy-mm-dd'),to_char(d,'d') dayofweek,to_char(d,'WW') WW,to_char(d,'IW') IW    from (    
select to_date('20131229','yyyymmdd')+level-1 as d from dual connect by level<=10);
TO_CHAR(D, D WW IW           
---------- - -- --
2013-12-29 1 52 52
2013-12-30 2 52 01
2013-12-31 3 53 01
2014-01-01 4 01 01
2014-01-02 5 01 01
2014-01-03 6 01 01
2014-01-04 7 01 01
2014-01-05 1 01 01
2014-01-06 2 01 02
2014-01-07 3 01 02

已选择10行。


对比日历,发现2013年12月30,31日的IW,均划分到了2014年的第一周

MONTH  YWE MWEEK   SUN   MON   TUE   WED   THU   FRI   SAT
------ --- ----- ----- ----- ----- ----- ----- ----- -----
12月   49      1     1     2     3     4     5     6     7
       50      2     8     9    10    11    12    13    14
       51      3    15    16    17    18    19    20    21
       52      4    22    23    24    25    26    27    28
       53      5    29    30    31

1)ww的算法为每年1月1日为第一周开始,date+6为每一周结尾

  例如20050101为第一周的第一天,而第一周的最后一天为20050101+6=20050107

  公式 每周第一天 :date + 周 * 7 - 7

  每周最后一天:date + 周 * 7 - 1

2)iw的算法为星期一至星期日算一周,且每年的第一个星期一为第一周,

   例如20050101为星期六,所以用iw的算法是前年的53周,而20050103之后才是第一周的开始。

  公式 每周第一天 :next_day(date) + 周 * 7 - 7

  每周最后一天:next_day(date) + 周 * 7 - 1


按照上述算法,WW的结果相差较远,而IW比较贴近我们对于日期的认识。
但是问题是IW存在边界问题。它会认为12月30日和31日是2014年的第一周,以此统计年历,则会出现问题。所以new_yweek解决的应该是这个问题。

可以使用下面的方法解决IW边界问题。

select case
           when (yweek = min(yweek)over(partition by mon order by yweek)) then
             mon_name
           else
             null
         end as month,
         yweek as yweek,
         row_number() over(partition by mon order by yweek) as mweek,
         sum(decode(wday, '1', mday, null)) as sun,
         sum(decode(wday, '2', mday, null)) as mon,
         sum(decode(wday, '3', mday, null)) as tue,
         sum(decode(wday, '4', mday, null)) as wed,
         sum(decode(wday, '5', mday, null)) as thu,
         sum(decode(wday, '6', mday, null)) as fri,
         sum(decode(wday, '7', mday, null)) as sat
    from (select dayofyear as everyday,
                 to_char(dayofyear, 'mm') as mon,
                 to_char(dayofyear, 'Month') as mon_name,
                 to_char(dayofyear, 'w') as mweek,
                 max(decode(to_char(dayofyear,'d'),'1',to_char(dayofyear, 'iw')+1,to_char(dayofyear, 'iw'))) over(order by dayofyear) as yweek,
                 to_char(dayofyear, 'd') as wday,
                 to_char(dayofyear, 'dd') as mday
            from (select to_date(&year || '0101', 'yyyymmdd') + level - 1 as dayofyear
                    from dual
                  connect by level <= to_char(to_date(&year || '1231', 'yyyymmdd'),'ddd')
                 )
         )
   group by mon, mon_name, yweek
  /


解决IW边界问题:
max(decode(to_char(dayofyear,'d'),'1',to_char(dayofyear, 'iw')+1,to_char(dayofyear, 'iw'))) over(order by dayofyear) as yweek
decode部分,如果日期是周日,则将iw的值+1,以便日历对齐。
而max分析函数部分,解决类似12月30日,31日划分到下一年的问题。
这个实现存在一些问题。
如果元旦是周五,周六或者周日,例如20110101,它是周六,IW会认为这天是2010年的第五十二周。

SQL> select to_char(to_date('20110101','yyyymmdd'),'IW') from dual;

TO
--
52
对于这个问题,我没有解决的方法,但是一个同事有另外一个方案,用自定义的周序列,使用偏移量。

select case
           when (yweek = min(yweek)over(partition by mon order by yweek)) then
             mon_name
           else
             null
         end as month,
         yweek as yweek,
         row_number() over(partition by mon order by yweek) as mweek,
         sum(decode(wday, '1', mday, null)) as sun,
         sum(decode(wday, '2', mday, null)) as mon,
         sum(decode(wday, '3', mday, null)) as tue,
         sum(decode(wday, '4', mday, null)) as wed,
         sum(decode(wday, '5', mday, null)) as thu,
         sum(decode(wday, '6', mday, null)) as fri,
         sum(decode(wday, '7', mday, null)) as sat
    from (select dayofyear as everyday,
                 to_char(dayofyear, 'mm') as mon,
                 to_char(dayofyear, 'Month') as mon_name,
                 to_char(dayofyear, 'w') as mweek,
                 ceil(to_number(dayIndex)/7) as yweek,
                 to_char(dayofyear, 'd') as wday,
                 to_char(dayofyear, 'dd') as mday
            from (
                    select
                    to_date(&year || '0101', 'yyyymmdd') + level - 1 as dayofyear,
                    to_char(s.firstday,'d')+rownum dayIndex
                    from
                    dual,(select to_date(&year || '0101','yyyymmdd')-1 firstday from dual) s
                    connect by level <= to_char(to_date(&year || '1231', 'yyyymmdd'),'ddd')
                 )
         )
   group by mon, mon_name, yweek
  /
   
case when   (to_char(to_date(&year || '0101', 'yyyymmdd'), 'd') > '1')
            and
            (to_char(dayofyear, 'd') < to_char(to_date(&year || '0101', 'yyyymmdd'), 'd'))
     then
            to_char(to_char(dayofyear, 'ww') + 1, 'fm00')
     else
            to_char(dayofyear, 'ww')
end as new_yweek
以2013年一月为例,元旦是周二(不是周日,满足了Case When的第一个条件)
6,7日是周日、周一,小于元旦的周二,所以所在周+1,那么行列转置后在日历上就下沉了一层。






来自 “ ITPUB博客 ” ,链接:http://blog.itpub.net/29254281/viewspace-776350/,如需转载,请注明出处,否则将追究法律责任。

转载于:http://blog.itpub.net/29254281/viewspace-776350/

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 以下是 SQL Server 生成全年日历的示例代码: ``` DECLARE @StartDate DATE = '2022-01-01'; DECLARE @EndDate DATE = '2022-12-31'; WITH dates AS ( SELECT @StartDate AS date UNION ALL SELECT DATEADD(day, 1, date) FROM dates WHERE DATEADD(day, 1, date) <= @EndDate ) SELECT DATEPART(year, date) AS [Year], DATEPART(month, date) AS [Month], DATEPART(day, date) AS [Day], DATENAME(weekday, date) AS [Weekday] FROM dates OPTION (MAXRECURSION 0); ``` 在上面的代码中,我们首先定义了开始和结束日期,然后使用递归CTE生成从开始日期到结束日期的所有日期。最后,我们选择每个日期的年份、月份、日份和星期几,并使用DATENAME函数将星期几转换为文本。选项(MAXRECURSION 0)指定在生成日期时不限制递归的次数。 执行以上代码,你将得到一个包含所有日期的结果集,其中包括每个日期的年份、月份、日份和星期几。 ### 回答2: 在SQL Server中生成全年日历可以通过以下步骤实现: 1. 创建一个日期表:首先,我们需要创建一个日期表,用来存储全年的日期信息。这个表可以包含日期、星期、月份等相关字段。可以使用日期函数来生成这个表的数据,例如使用`DATEADD`函数来增加日期,`DATENAME`函数来获取星期和月份的名称。 2. 插入数据:使用`INSERT INTO`语句将生成日期数据插入到日期表中。可以使用循环或递归的方式逐行插入数据。在插入数据之前,可以使用`TRUNCATE TABLE`语句清空原有的日期表,以免出现重复数据。 3. 定义日期范围:根据需要生成日历范围,可以使用`DECLARE`语句定义起始日期和结束日期的变量。例如:`DECLARE @StartDate DATE = '2022-01-01', @EndDate DATE = '2022-12-31'`。 4. 生成日历:使用`SELECT`语句查询日期表,并使用`WHERE`子句过滤出在日期范围内的数据。可以按照一周七天或一个月的方式进行分组,并使用`ORDER BY`子句按照日期的顺序排序。 下面是一个示例的SQL查询,用来生成2022年度的日历: ``` DECLARE @StartDate DATE = '2022-01-01', @EndDate DATE = '2022-12-31' CREATE TABLE #Calendar ( DateColumn DATE, WeekdayColumn VARCHAR(20), MonthColumn VARCHAR(20) ) WHILE @StartDate <= @EndDate BEGIN INSERT INTO #Calendar (DateColumn, WeekdayColumn, MonthColumn) VALUES (@StartDate, DATENAME(WEEKDAY, @StartDate), DATENAME(MONTH, @StartDate)) SET @StartDate = DATEADD(DAY, 1, @StartDate) END SELECT WeekdayColumn, DateColumn, MonthColumn FROM #Calendar WHERE DateColumn BETWEEN @StartDate AND @EndDate ORDER BY DateColumn ``` 以上就是使用SQL Server生成全年日历的一种方法。可以根据自己的需要进行调整和扩展,来满足不同的日历需求。 ### 回答3: 要在SQL Server中生成全年日历,可以使用递归查询和日期函数来实现。以下是一种实现方法: 首先,创建一个表来存储所有的日期和相关的信息,包括年份、月份、日期、星期、是否为工作日等等。 ```sql CREATE TABLE Calendar ( Year INT, Month INT, Day INT, Weekday INT, IsWorkingDay BIT ); ``` 接下来,使用递归查询来生成所有的日期。递归查询就是在查询的过程中调用自己来进行迭代操作。 ```sql WITH RecursiveCalendar AS ( -- 基础查询,生成第一个日期 SELECT DATEPART(YEAR, GETDATE()) AS Year, DATEPART(MONTH, GETDATE()) AS Month, DATEPART(DAY, GETDATE()) AS Day, DATEPART(WEEKDAY, GETDATE()) AS Weekday, CASE WHEN DATEPART(WEEKDAY, GETDATE()) NOT IN (1, 7) THEN 1 ELSE 0 END AS IsWorkingDay UNION ALL -- 递归查询,生成后续日期 SELECT Year, Month, Day+1, (Weekday+1) % 7, CASE WHEN (Weekday+1) % 7 NOT IN (1, 7) THEN 1 ELSE 0 END FROM RecursiveCalendar WHERE Day+1 <= DATEPART(DAY, DATEFROMPARTS(Year, Month+1, 1)) ) -- 将结果插入到Calendar表中 INSERT INTO Calendar (Year, Month, Day, Weekday, IsWorkingDay) SELECT Year, Month, Day, Weekday, IsWorkingDay FROM RecursiveCalendar OPTION (MAXRECURSION 366); -- 设置递归最大次数为366,确保处理闰年 ``` 最后,你就可以使用以下查询来获取全年的日历数据: ```sql SELECT * FROM Calendar; ``` 以上就是在SQL Server中生成全年日历的方法。通过使用递归查询和日期函数,我们可以轻松地生成任何年份的日历数据。请注意,上述示例仅提供了基础实现,你可以根据自己的需求对其进行修改和优化。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值