SQL Server获取连续区间的日期

摘要:
当前的实现方法包括:master..Spt_Get值。使用WHILE循环获取光标。获取CTE递归。(感谢您提供)方法1:使用系统表主表.Spt_Values获取1.获取连续天数--获取连续日期DECLARE@StartTimeDATE='2019-03-08',--开始时间@EndTimeDATE='2019-03-18'--结束时间SELECTCONVERT(NVARCH

目前实现方法有:

  1. 通过系统表master..spt_values获取
  2. 用WHILE循环获取
  3. 游标获取
  4. CTE递归(感谢博友提供)

方法一:通过系统表master..spt_values获取

1、获取连续的日

-- 获取连续区间的日期
DECLARE @StartTime DATE = '2019-03-08', -- 开始时间
        @EndTime   DATE = '2019-03-18' -- 结束时间

SELECT CONVERT(NVARCHAR(10),DATEADD(DAY,number,@StartTime),120) AS DayTime
  FROM master..spt_values  
 WHERE type = 'p' 
   AND number <= DATEDIFF(DAY,@StartTime,@EndTime)

SQL Server获取连续区间的日期第1张

 
2、获取连续的月

-- 获取连续区间的月份
DECLARE @StartTime DATE = '2019-03-08', -- 开始时间
        @EndTime   DATE = '2019-12-28' -- 结束时间

SELECT CONVERT(VARCHAR(7),DATEADD(MONTH,number,@StartTime),120) AS MonthTime
  FROM master..spt_values WITH (NOLOCK) 
 WHERE type='P' 
   AND number <= DATEDIFF(MONTH,@StartTime,@EndTime)

SQL Server获取连续区间的日期第2张

3、获取连续的年

-- 获取连续区间的年份
DECLARE @StartTime DATE = '2019-03-08', -- 开始时间
        @EndTime   DATE = '2020-03-08' -- 结束时间

SELECT CONVERT(VARCHAR(4),DATEADD(YEAR,number,@StartTime),120) AS YearTime
  FROM master..spt_values WITH (NOLOCK) 
 WHERE type='P' 
   AND number <= DATEDIFF(YEAR,@StartTime,@EndTime)

SQL Server获取连续区间的日期第3张

master..spt_values原理:

通过number来实现年月日的加减,因为number值最大是2047,所以只能连续加2047。

SQL Server获取连续区间的日期第4张

如图:结束时间是'2026-03-28',但我们降序可以看到,从2019-03-08只能增加连续2047天,如此2024-10-24之后的日期就没有了。

所以如果连续区间超过了2048,则此方法不适用,此时可以用以下方法二来实现

方法二:用WHILE循环获取

1、获取连续的日

-- 连续获取天
DECLARE @StartTime DATE = '2019-03-08', -- 开始时间
        @EndTime   DATE = '2019-03-18' -- 结束时间

-- 创建临时表#DateTime存储日期
CREATE TABLE #DateTime
(
    DayTime DATE
);

-- 循环获取日期插入临时表
WHILE @StartTime <= @EndTime
BEGIN
    INSERT INTO #DateTime (DayTime)
    VALUES (@StartTime);
    SET @StartTime = DATEADD(DAY, 1, @StartTime);
END;

SELECT DayTime FROM #DateTime;

-- 删除临时表
DROP TABLE #DateTime;

SQL Server获取连续区间的日期第5张

2、获取连续的月

-- 连续获取的月
DECLARE @StartTime DATE = '2019-03-08', -- 开始时间
        @EndTime   DATE = '2019-12-28' -- 结束时间

-- 创建临时表#MonthTime存储日期
CREATE TABLE #MonthTime
(
    MonthTime VARCHAR(7)
);



-- 循环获取日期插入临时表
WHILE @StartTime <= @EndTime
BEGIN
    INSERT INTO #MonthTime (MonthTime)
    VALUES (CONVERT(VARCHAR(7),@StartTime));
    SET @StartTime = DATEADD(MONTH, 1, @StartTime);
END;

SELECT MonthTime FROM #MonthTime;

-- 删除临时表
DROP TABLE #MonthTime;

SQL Server获取连续区间的日期第6张

3、获取连续的年

-- 连续获取的月
DECLARE @StartTime DATE = '2019-03-08', -- 开始时间
        @EndTime   DATE = '2025-03-28' -- 结束时间

-- 创建临时表#YearTime存储日期
CREATE TABLE #YearTime
(
    YearTime VARCHAR(4)
);



-- 循环获取日期插入临时表
WHILE @StartTime <= @EndTime
BEGIN
    INSERT INTO #YearTime (YearTime)
    VALUES (CONVERT(VARCHAR(4),@StartTime));
    SET @StartTime = DATEADD(YEAR, 1, @StartTime);
END;

SELECT YearTime FROM #YearTime;

-- 删除临时表
DROP TABLE #YearTime;

SQL Server获取连续区间的日期第7张

方法三:用游标获取,原理和WHILE循环相似,此处不再展示

方法四:评论区博友说的CTE递归

DECLARE @StartTime DATE = '2018-12-08', -- 开始时间
@EndTime DATE = '2019-03-18' -- 结束时间
;
-- 获取连续天
WITH CteDateDay AS
(
SELECT @StartTime DayTime
UNION ALL
SELECT DATEADD(DAY,1,DayTime) DayTime FROM CteDateDay
WHERE DayTime<@EndTime
)
SELECT DayTime FROM CteDateDay
OPTION (MAXRECURSION 0)
;

-- 获取连续月
WITH CteDateMonth AS
(
SELECT CONVERT(VARCHAR(7),@StartTime,120) MonthTime
UNION ALL
SELECT CONVERT(VARCHAR(7),DATEADD(MONTH,1,CAST(MonthTime+'-01' AS DATE)),120) DayTime FROM CteDateMonth
WHERE MonthTime<CONVERT(VARCHAR(7),@EndTime,120)
)
SELECT MonthTime FROM CteDateMonth
OPTION (MAXRECURSION 0)
;

-- 获取连续年
WITH CteDateYear AS
(
SELECT DATEPART(YEAR,@StartTime) YearTime
UNION ALL
SELECT YearTime+1 DayTime FROM CteDateYear
WHERE YearTime<DATEPART(YEAR,@EndTime)
)
SELECT YearTime FROM CteDateYear
OPTION (MAXRECURSION 0)

 

SQL Server获取连续区间的日期第8张

如果有什么好的方法和建议,欢迎大家来指点,谢谢!

免责声明:文章转载自《SQL Server获取连续区间的日期》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇Ubuntu 安装和卸载sublime【TOMCAT】Tomcat gzip压缩传输数据下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

Mysql 查询优化,索引原理与慢查询优化。

Mysql 查询优化 索引原理与慢查询优化 一 介绍 为什么要有索引? 一般的应用程序,读写比例在10:左右,而且插入操作和一般的更新操作很少出现性能问题, 在生产环境中,我们遇到最多的,也是最容易出问题的,还是一些复杂的查询操作,因此对查询, 我们必定要去优化它们,那我们怎么去优化呢?那就要提到索引了。 什么是索引? 索引在MySQL中也叫做“...

js new Date()参数格式

最近在写页面使用new Date()获取时间戳在ie浏览器中测试发现无效;后来发现是参数格式问题, new Date()参数格式如下: 1、用整数初始化日期对象var date1 = new Date(2017,06,06); console.log(date1); // Thu Jul 06 2017 00:00:00 GMT+0800 (中国标准时间)...

2-14-1 MySQL基础语句,查询语句

一. SQL概述 结构化查询语言(Structured Query Language)简称SQL 1. 它是一种特殊目的的编程语言 2. 它还是一种数据库查询和程序设计语言 (用于存取数据以及查询、更新和管理关系数据库系统;同时也是数据库脚本文件的扩展名) 从上可以看出我们数据库相关工作职位大概两种:DBD和DBA SQL 是1986年10 月由美国国家标...

Mysql源码学习——打造专属语法

语法分析——YACC 接触过SQL语句的人都会看过这家或者那家的SQL手册,其语法标准应该是从SQL92开始吧,在看SQL92标准的时候,你会发现里面定义的都是一些巴科斯范式(BNF),就是一种语法定义的标准。不管是牛X哄哄的ORACLE,还是不幸被其收购的Mysql,都会遵循里面的标准语法,当然一些扩展的语法除外,比如今天我们就会扩展一个简单的语法^...

pg10、11、12版本特性

2. 分区表的改进 PostgreSQL 10 实现了声明式分区,PostgtreSQL 11完善了功能,PostgreSQL 12提升了性能。我们知道在PostgreSQL 9.X时代需要通过表继承实现分区,这时还需要手工加触发器或规则把新插入的数据重新定向到具体的分区中,从PostgreSQL 10之后不需要这样了,直接用声明式分区就可以了,语法如下:...

mysql中的view(虚拟表)

视图是从一个或多个表/视图中导出来的虚拟表。mysql支持可更新的视图。 1 创建 CREATE [OR REPLACE] ##使用这个选项,若视图已存在,则等图create,若视图不存在,则等同alter [ALGORITHM = {UNDEFINED | MERGE | TEMPTABLE}] [DEFINER = {...