sql server 自定义分割月功能详解及实现代码

在最近的项目开发过程中,遇到了Sql server自动分割月的功能需求,这里在网上整理下资料.

1、为何出现自定义分割月的需求

今天梳理一个平台的所有函数时,发现了一个自定义分割月函数,也就是指定分割月的开始日索引值(可以从1-31闭区间内的任何一个值)来获取指定日期所对应的分割月数值。这个函数当时是为了解决业务部门获取非标准月(标准月就是从每个月的第一天到最后一天组成一个完成的标准月份)的统计汇总数据的。例如:如果指定分割月的开始日索引值为5则表示某个月的5号到下个月的4号之间作为一个完整的分割月;同样地如果指定分割月的开始日索引值为1则表示标准月等等。

我仔细梳理了这个函数进行了重构简化以及扩展,该自定义分割月函数的实现区别之前写的SQL Server时间粒度系列----第3节旬、月时间粒度详解文章中将一个整数值和月份日期相互转换功能,这个是按照标准月来实现的,虽然思路大致相同,但是并没有针对之前的月份日期和整数值转换函数对来进行扩展而是独立开发新的功能函数。也是为了尽量做到函数功能职责单一性、稳定性、可维护性以及可扩展性。

2、sql server实现自定义分割月功能

自定义分割月功能函数包括两个标量函数:ufn_SegMonths和ufn_SegMonth2Date。ufn_SegMonths获取指定的日期在自定义分割月对应的分割月数值;ufn_SegMonth2Date获取指定一个分割月数值赌对应的月份日期。

sql server 版本的实现T-SQL代码如下:

IF OBJECT_ID(N'[dbo].[ufn_SegMonths]', 'FN') IS NOT NULL BEGIN DROP FUNCTION [dbo].[ufn_SegMonths]; END GO --================================== -- 功能:根据自定义月开始索引值获取指定日期所在的自定义月数。 -- 说明:自定义分割月数 = 年整数值*100 + 当前所在分割月值。 -- 环境:SQL Server 2005+。 -- 调用:SET @intSegMonths = dbo.fn_SegMonths('2008-01-14', 15)。 -- 创建:XXXX-XX-XX XX:XX-XX:XX XXX 创建函数实现。 -- 修改:XXXX-XX-XX XX:XX-XX:XX XXX XXXXXXXX。 --================================== CREATE FUNCTION [dbo].[ufn_SegMonths] ( @dtmDate AS DATETIME -- 日期 ,@tntSegStartIndexOfMonth AS INT = 15 -- 自定义分割月开始索引值(1-31) ) RETURNS INT AS BEGIN IF (@tntSegStartIndexOfMonth = 0 OR @tntSegStartIndexOfMonth >= 32) BEGIN SET @tntSegStartIndexOfMonth = 15; END DECLARE @intYears AS INT ,@tntMonth AS TINYINT ,@sntDay AS SMALLINT; SELECT @intYears = DATEDIFF(YEAR, '1900-01-01', @dtmDate) ,@tntMonth = DATEPART(MONTH, @dtmDate) ,@sntDay = DATEPART(DAY, @dtmDate); IF (@sntDay >= @tntSegStartIndexOfMonth) BEGIN SET @tntMonth = @tntMonth + 1; END IF (@tntMonth > 12) BEGIN SELECT @intYears = @intYears + 1 ,@tntMonth = @tntMonth - 12; END RETURN @intYears * 100 + @tntMonth; END GO IF OBJECT_ID(N'[dbo].[ufn_SegMonths2Date]', 'FN') IS NOT NULL BEGIN DROP FUNCTION [dbo].[ufn_SegMonths2Date]; END GO --================================== -- 功能:获取自定义分割月数对应的自定义分割月日期。 -- 说明:自定义分割月日期 = 自定义分割月数/100对应的年整数日期“组合”当前所在分割月值。 -- 环境:SQL Server 2005+。 -- 调用:SET @dtmSegMonthDate = dbo.fn_SegMonths2Date(11602)。 -- 创建:XXXX-XX-XX XX:XX-XX:XX XXX 创建函数实现。 -- 修改:XXXX-XX-XX XX:XX-XX:XX XXX XXXXXXXX。; --================================== CREATE FUNCTION [dbo].[ufn_SegMonths2Date] ( @intSegMonths AS INT -- 自定义分割月数 ) RETURNS DATETIME AS BEGIN DECLARE @dtmDefaultBasedate AS DATETIME; SET @dtmDefaultBasedate = '1900-01-01'; IF ((@intSegMonths IS NULL) OR (@intSegMonths <= 0)) BEGIN RETURN @dtmDefaultBasedate; END DECLARE @intYears AS INT ,@intMonth AS INT; SELECT @intYears = @intSegMonths / 100 ,@intMonth = @intSegMonths % 100; RETURN DATEADD(MONTH, @intMonth - 1, DATEADD(YEAR, @intYears, @dtmDefaultBasedate)); END GO

3、测试验证效果

针对以上简单的测试代码如下:

DECLARE @dtmStartDate AS DATETIME ,@dtmEndDate AS DATETIME; SELECT @dtmStartDate = '2000-01-01' ,@dtmEndDate = '2016-12-31'; SELECT [T1].* ,[dbo].[ufn_SegMonths2Date]([T1].[SegMonths]) AS SegMonthDate FROM ( SELECT [T].[CDate] ,[dbo].[ufn_SegMonths]([T].[CDate], 28) AS SegMonths FROM ( SELECT DATEADD(DAY, [Num], @dtmStartDate) AS CDate FROM [dbo].[ufn_GetNums](0, DATEDIFF(DAY, @dtmStartDate, @dtmEndDate)) ) AS T WHERE [T].[CDate] BETWEEN '2014-12-01' AND '2016-03-31' ) AS T1 WHERE DATEPART(DAY, [T1].[CDate]) >= 27 GO

效果截图如下:

注意:以上测试代码使用了SQL Server数字辅助表的实现这边文章的内联表值函数ufn_GetNums。

4、总结语

这次是梳理平台的功能性函数所进行的重构简化以及扩展的实现。尽量将日期有关的功能函数梳理出来,便于直接在sql server用户数据库中来使用, 也便于BI仓库中使用。国庆一来已经过去一周,原来打算一周一遍的计划还是延期啦,再次严重检讨自己。

感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

时间: 2024-10-22 02:44:18

sql server 自定义分割月功能详解及实现代码的相关文章

sql server 自定义分割月功能详解及实现代码_MsSql

在最近的项目开发过程中,遇到了Sql server自动分割月的功能需求,这里在网上整理下资料.       1.为何出现自定义分割月的需求 今天梳理一个平台的所有函数时,发现了一个自定义分割月函数,也就是指定分割月的开始日索引值(可以从1-31闭区间内的任何一个值)来获取指定日期所对应的分割月数值.这个函数当时是为了解决业务部门获取非标准月(标准月就是从每个月的第一天到最后一天组成一个完成的标准月份)的统计汇总数据的.例如:如果指定分割月的开始日索引值为5则表示某个月的5号到下个月的4号之间作为

SQL Server 2012 FileTable 新特性详解

FileTable是基于FILESTREAM的一个特性.有以下一些功能: •一行表示一个文件或者目录. •每行包含以下信息: • •file_Stream流数据,stream_id标示符(GUID). •用户表示和维护文件及目录层次关系的path_locator和parent_path_locator •有10个文件属性 •支持对文件和文档的全文搜索和语义搜索的类型列. •filetable强制执行某些系统定义的约束和触发器来维护命名空间的语义 •针对非事务访问时,SQL Server配置FIL

基于SQL Server OS的任务调度机制详解_MsSql

简介      SQL Server OS是在Windows之上,用于服务SQL Server的一个用户级别的操作系统层次.它将操作系统部分的功能从整个SQL Server引擎中抽象出来,单独形成一层,以便为存储引擎提供服务.SQL Server OS主要提供了任务调度.内存分配.死锁检测.资源检测.锁管理.Buffer Pool管理等多种功能.本篇文章主要是谈一谈SQL OS中所提供的任务调度机制. 抢占式(Preemptive)调度与非抢占式(non-Preemptive)调度     数据

SQL SERVER 2000安装教程图文详解_MsSql

注意:Windows XP不能装企业版.win2000\win2003服务器安装企业版一.硬件和操作系统要求 下表说明安装 Microsoft SQL Server 2000 或 SQL Server 客户端管理工具和库的硬件要求. 硬件 最低要求计算机 Pentium 166 MHz 或更高. 内存 (RAM)至少 64 MB,建议 128 MB 或更多.根据笔者的经验,内存容量可以和数据容量保持1:1的比例,这样可以更好的发挥其效能.硬盘空间需要约500MB的程序空间,以及预留500M的数据

sql server 常用的扩展存储过程详解

sql server 里面提供了丰富的系统存储过程来辅助我们管理数据库以及开发.今天分享介绍一些常用的数据库扩展存储过程  xp_cmdshell 这个大家都比较熟悉了,使用xp_cmdshell 可以在连接sql server 的时候可以执行doc 命令 最简单一个例子EXEC master.sys.xp_cmdshell 'dir D:\' 获取D 盘下面的目录.当然这个D:\ 是服务器上面的,不是本地电脑的. 所以有时候可以使用 xp_cmdshell 配合作业都角度使用执行bcp等命令来

SQL SERVER 2000安装教程图文详解

注意:Windows XP不能装企业版.win2000\win2003服务器安装企业版 一.硬件和操作系统要求 下表说明安装 Microsoft SQL Server 2000 或 SQL Server 客户端管理工具和库的硬件要求. 硬件 最低要求计算机 Pentium 166 MHz 或更高. 内存 (RAM)至少 64 MB,建议 128 MB 或更多.根据笔者的经验,内存容量可以和数据容量保持1:1的比例,这样可以更好的发挥其效能.硬盘空间需要约500MB的程序空间,以及预留500M的数

sql server的cube操作符使用详解

server|详解 cube操作符要使用cube,首先要了解group by其实cube和rollup区别不太大,只是在基于group by 子句创建和汇总分组的可能的组合上有一定差别,cube将返回的更多的可能组合.如果在 group by 子句中有n个列或者是有n个表达式的话,sqlserver在结果集上会返回2的n-1次幂个可能组合.注意:使用cube操作符时,最多可以有10个分组表达式在cube中不能使用all关键字例子:我们在数据库统计中常常要查询以下情况:如一个定单数据库,我们要知道

SQL Server 2008特性及集成服务详解

SQL Server 2008系统诞生于08年3月13日,在微软2008新一代http://www.aliyun.com/zixun/aggregation/13760.html">企业应用平台与开发技术发布大会上,微软宣布向企业用户大众同时发布三款核心应用平台产品:Windows Server 2008.Visual Studio 2008.SQL Server 2008,此次微软发布的三大产品对企业实现"动态IT"愿景.随着Windows Server 2008.V

SQL Server中BUILTIN\Administrators用户详解说明

SQL Server 是一个http://www.aliyun.com/zixun/aggregation/22.html">关系数据库管理系统.它最初是由Microsoft Sybase 和Ashton-Tate三家公司共同开发的,于1988 年推出了第一个OS/2 版本.在Windows NT 推出后,Microsoft与Sybase 在SQL Server 的开发上就分道扬镳了,Microsoft 将SQL Server 移植到Windows NT系统上,专注于开发推广SQL Ser