使用DATEADD和DATEDIFF来计算SQL Server的DATETIME值

 

在SQL Server数据库中,DATETIME和SMALLDATETIME值是以整数存储的。然而,与整数不同的是,它们不能直接地进行数学运算。尽管如此,有时候还是需要在日期/时间值中添加或减去一个时间间隔。比如,你可能想在一值上加一些月数或天数,或者甚至可能是小时数。你甚至可能想比较两个日期/时间值以便确定它们之间的时间间隔,如相差的天数或年数。为了简化这些类型的计算,Transact-SQL支持两个重要的日期/时间方法:DATEADD和DATEDIFF。

 

在关于DATETIME值这一系列文章的第四部分,我阐述了如何使用这两个方法并举例说明它们是如何工作的。为了演示这些方法,我使用了下面的Transact-SQL代码在AdventureWorks示例数据库中创建了一个Sales.Orders表:

 


USE AdventureWorks
GO
IF EXISTS (SELECT table_name 
FROM information_schema.tables 
WHERE table_schema = 'Sales'
AND table_name = 'Orders')
DROP TABLE Sales.Orders
GO
CREATE TABLE Sales.Orders
(
OrderID INT NOT NULL,
OrderDate DATETIME NOT NULL,
DelivDate DATETIME NOT NULL

GO
INSERT INTO Sales.Orders
VALUES(1001, GETDATE(), '2008-09-08 18:27:10.750')

 

表的定义包含了OrderDate和DelivDate字段,两者都是DATETIME数据类型。在我创建了表之后,我在表中插入了一行用于测试DATEADD和DATEDIFF方法的数据。

 

使用DATEADD方法

 

在一些情况下,你可能想添加一个时间间隔到DATETIME或SMADDDATETIME值中——或者减去一个时间间隔。比如,你可能需要在一个指定的日期中增加或减去一个月。你可以使用DATEADD方法来执行这个计算。这个方法运用了下面的语法:

 


DATEADD(<date/time_part>, <number>, <date>)

 

<date/time_part>占位符指的是日期/时间值中增加或减少的增量/余差(如日或月)。下表列出了可以使用的日期/时间部分,以及代表这些部分的缩写:

 

Date/time part

Abbreviations

year

yy, yyyy

quarter

qq, q

month

mm, m

day of year

dy, y

day

dd, d

week

wk, ww

weekday

dw

hour

hh

minute

mi, n

second

ss, s

millisecond

ms

 

比如,如果你想在一个日期/时间值中增加一小时,可以使用hh缩写。在某些情况下,日期/时间部分支持两个缩写,如周可以用wk或ww支持。

 

<number>占位符指的是所增加的数值(一个整数)。比如,如果在日期中增加10天,就是10。但是,注意,如果是减去时间间隔,它必须是一个负整数。比如,从天数中减去10,就必须是-10。

 

<date>占位符指的是增加或减少的指定间隔的日期/时间值。它可能是一个日期/时间格式的字符串值,或者是方法返回的一个日期/时间值,又或者是常见的DATETIME或SMALLDATETIME字段。

 

让我们举例来说明它是如何工作的。在下面的SELECT语句中,我增加三个月到Sales.Orders表中OrderDate值:

 


SELECT OrderDate, DATEADD(mm, 3, OrderDate) AS NewDate
FROM Sales.Orders
WHERE OrderID = 1001

 

注意,SELECT列表使用了DATEADD方法。这个方法有三个参数:mm指月,3指月数,而OrderDate是一个DATETIME值。因此,当查询返回值时,每个OrderDate值都会增加三个月时间,如下的结果所示:

 

OrderDate

NewDate

2008-08-27 13:36:16.280

2008-11-27 13:36:16.280

 

如上所示,日期August 27已经被改为November 27。而且,这样的运算还不仅限于日期。下面我在OrderDate值中增加三个小时:

 


SELECT OrderDate, DATEADD(hh, 3, OrderDate) AS NewTime
FROM Sales.Orders
WHERE OrderID = 1001

 

DATEADD的第一个参数现在是hh,而不是mm,因此,只有小时被改变了,如下结果所示:

 

OrderDate

NewTime

2008-08-27 13:36:16.280

2008-08-27 16:36:16.280

 

日期/时间值也可以减去一定的日期或时间间隔。在下例中,我从OrderDate值中减去了三天:

 


SELECT OrderDate, DATEADD(dd, -3, OrderDate) AS PastDate
FROM Sales.Orders
WHERE OrderID = 1001

 

注意,DATEADD的第一个参数现在是dd。同时,注意,第二个参数是一个负数,这意味着将有三天被减去,如下所示:

 

OrderDate

PastDate

2008-08-27 13:36:16.280

2008-08-24 13:36:16.280

 

这样,新的日期是August 24而不是August 27。

 

这样,上面的例子演示如何在从数据库查询到日期/时间值后再对它进行修改。而DATEADD方法同样也可以用来插入日期/时间数据。因为DATEADD方法返回一个DATETIME值。(如果所提供的日期对应的方法是SMALLDATETIME,那么它将返回一个SMALLDATETIME值。)在下面的例子中,我添加了一行数据到Sales.Orders表中,然后使用SELECT语句来检索这个行:

 


INSERT INTO Sales.Orders
VALUES(1002, GETDATE(), DATEADD(dd, 10, GETDATE()))
GO
SELECT * FROM Sales.Orders
WHERE OrderID = 1002

 

注意,VALUES子句包含了表中每个字段的值。对于OrderDate值,我使用GETDATE()方法来获取当前的日期和时间。对于DelivDate字段,我使用DATEADD方法以及相应的三个参数。第一个参数dd表示将要添加到日期中的是天数。第二个参数10意味着将添加10天到日期中。最后,第三个参数是GETDATE方法。因此,10天将添加到目前的日期和时间中并插入到DelivDate字段。这就是SELECT语句生成的结果:

 

OrderID

OrderDate

DelivDate

1002

2008-08-27 13:40:22.357

2008-09-06 13:40:22.357

 

正如所期待的,DelivDate值比OrderDate晚10天。

 

现在让我们来检测一个使用了DATEADD方法的UPDATE语句。在下面的语句中,我从DelivDate值中减去了三天,然后显示了结果:

 


UPDATE Sales.Orders
SET DelivDate = DATEADD(dd, -3, DelivDate)
WHERE OrderID = 1002
GO
SELECT * FROM Sales.Orders
WHERE OrderID = 1002

 

这次我在SET子句中使用了DATEADD——我将DelivDate值设为DATEADD方法返回的结果。这个方法指定天数(dd)为第一个参数,-3为第二个参数,而DelivDate字段为第三个参数。这就意味着该方法将返回一个比原始日期早三天的日期,并将DelivDate设置为新的日期,如下结果显示:

 

OrderID

OrderDate

DelivDate

1002

2008-08-27 13:40:22.357

2008-09-03 13:40:22.357

 

你应该记得,INSERT语句(在前一个例子)添加了一个DelivDate值为September 6的行。但是,这个值现在是September 3,比原来早了三天。

使用DATEDIFF方法

 

DATEDIFF方法可以计算两个日期之间的时间间隔,并返回一个代表间隔的整数。这个方法使用下面的语法:

 


DATEDIFF(<date/time_part>, <start_date>, <end_date>)

 

<date/time_part>占位符指的是两个日期中需要比较的部分。比如,你想确认开始日期和结束日期之间的小时数或天数。

 

除了工作日(dw, w)缩写之外,<date/time_part>占位符使用的缩写与DATEADD方法一样。DATEDIFF不支持工作日比较。

 

<start_date>占位符指的是比较的开始日期,而<end_date>占位符指的是结束日期。换言之,方法将返回开始日期和结束日期之间的具体时间或日期间隔。

 

让我们举例来说明它是如何工作的。下面的SELECT语句计算了Sales.Orders 表中OrderDate和DelivDate值之间的时间间隔:

 


SELECT OrderDate, DelivDate, 
DATEDIFF(dd, OrderDate, DelivDate) AS DaysDiff
FROM Sales.Orders
WHERE OrderID = 1002

 

在这个语句中,我在SELECT列表中使用了DATEDIFF。这个方法的第一参数指定间隔必须是天数(dd),而第二个参数指定OrderDate作为开始日期,然后第三个参数指定DelivDate作为结束日期。因此,DATEDIFF将计算OrderDate和DelivDate之间的天数,在此例子中,它是7天,如下结果所示:

 

OrderDate

DelivDate

DaysDiff

2008-08-27 13:40:22.357

2008-09-03 13:40:22.357

7

 

当然,它也可以用来计算各种时间间隔,如下例的语句所示:

 


SELECT OrderDate, DelivDate, 
DATEDIFF(hh, OrderDate, DelivDate) AS HoursDiff
FROM Sales.Orders
WHERE OrderID = 1002

 

在这种情况下,方法的第一个参数是小时(hh)而非天数。因此,方法将返回OrderDate 和DelivDate值之间相差的小时数,如下结果所示:

 

OrderDate

DelivDate

HoursDiff

2008-08-27 13:40:22.357

2008-09-03 13:40:22.357

168

 

两个值之间相差168个小时。

 

与DATEADD方法一样,DATEDIFF方法并不仅限于用在SELECT语句中。比如,DATEDIFF可以用在UPDATE语句的WHERE子句中,以确定哪一行需要更新。在下例中,我使用了DATEDIFF来指定这些在OrderDate和DelivDate值之间天数少于8的行。

 


UPDATE Sales.Orders
SET DelivDate = DATEADD(dd, 3, DelivDate)
WHERE DATEDIFF(dd, OrderDate, DelivDate) < 8
GO
SELECT OrderID, OrderDate, DelivDate, 
DATEDIFF(dd, OrderDate, DelivDate) AS DaysDiff
FROM Sales.Orders

 

在前面的例子中,DATEDIFF方法返回了OrderDate和DelivDate值之间的天数。然后这个数目将与8作比较。如果天数少于8,那么这一行将被更新;否则,该行将不改变。对于这些需要更新的行,我使用DATEADD方法来增加三天到DelivDate值中。然后,我运行一个SELECT语句来返回Sales.Orders表的数据以及计算每个行中两个日期的不同,如下结果所示:

 

OrderID

OrderDate

DelivDate

DaysDiff

1001

2008-08-27 13:36:16.280

2008-09-08 18:27:10.750

12

1002

2008-08-27 13:40:22.357

2008-09-06 13:40:22.357

10

 

结果显示现在在两个日期(第二行中)之间是相差10天,而非原来的7天。

 

在表定义中使用DATEADD和DATEDIFF

 

DATEADD和DATEDIFF方法也可以用在表定义中。例如,字段定义的DEFAULT子句中可以用DATEADD方法或使用DATEDIFF方法来创建一个计算得来的字段。在下面的Transact-SQL代码中,我首先创建了使用DATEADD和DATEDIFF的表,然后添加一行数据到表中,最后检索表的数据:

 


USE AdventureWorks
GO
IF EXISTS (SELECT table_name 
FROM information_schema.tables 
WHERE table_schema = 'Sales'
AND table_name = 'Orders')
DROP TABLE Sales.Orders
GO
CREATE TABLE Sales.Orders
(
OrderID INT NOT NULL,
OrderDate DATETIME NOT NULL DEFAULT GETDATE(),
DelivDate DATETIME NOT NULL DEFAULT DATEADD(dd, 10, GETDATE()),
DaysDiff AS DATEDIFF(dd, OrderDate, DelivDate)

GO
INSERT INTO Sales.Orders(OrderID)
VALUES(1001)
GO
SELECT OrderID, OrderDate, DelivDate, DaysDiff
FROM Sales.Orders

 

在CREATE TABLE语句中,我创建了四个字段,其中三个存储日期/时间数据。OrderDate字段直接使用GETDATE来生成默认值。DelivDate字段也有一个默认值。然而,这个默认值是基于DATEADD返回的结果的,同时,在这种情况下,我使用方法增加10天到GETDATE返回的值存储到DelivDate字段中。最后,DaysDiff字段是一个计算得来的字段,它的值是使用DATADIFF来计算OrderDate和DelivDate值之间的天数差。

 

在表定义之后,我插入一数据行到表中。因为所有的日期/时间值都是自动生成的,因此我仅仅需要插入OrderID值,如下所示:

 

OrderID

OrderDate

DelivDate

DaysDiff

1001

2008-08-27 13:42:50.433

2008-09-06 13:42:50.433

10

 

DATEADD和DATEDIFF方法不仅仅在表定义中非常有用,同样也适用于查询和数据修改语句。通过DATEADD,我们可以将日期/时间值增加和减少一定值,而通过DATEDIFF,我们可以计算日期/时间值之间的时间间隔。更多详细的关于这些方法的信息,可以阅读Microsoft SQL Server Books Online。

你可能感兴趣的:(使用DATEADD和DATEDIFF来计算SQL Server的DATETIME值)