Как преобразовать DATETIME в значение FILETIME в T-SQL? - PullRequest
3 голосов
/ 27 мая 2010

Мне нужно преобразовать SQL Server DATETIME значение в FILETIME в операторе T-SQL SELECT (в SQL Server 2000). Есть ли встроенная функция для этого? Если нет, может ли кто-нибудь помочь мне понять, как реализовать эту процедуру преобразования в виде UDF (или просто Transact-SQL)? Вот что я знаю:

  1. FILETIME - это 64-битное значение, представляющее количество интервалов в 100 наносекунд с 1 января 1601 года (UTC) (на MSDN: FILETIME Структура ).
  2. Базовое время SQL Server начинается с 1900-01-01 до 00:00:00 (для SELECT CAST (0 в качестве DATETIME)).

Я нашел несколько примеров, показывающих, как преобразовать значения FILETIME в T-SQL DATETIME (хотя я не уверен на 100%, что они точны), но не смог найти ничего об обратном преобразовании. Даже общая идея (или алгоритм) поможет.

Ответы [ 2 ]

3 голосов
/ 01 июня 2010

Хорошо, я думаю, что смог реализовать это сам. Вот функция:

IF EXISTS 
(
    SELECT 1
    FROM   sysobjects 
    WHERE  id   = OBJECT_ID('[dbo].[fnDateTimeToFileTime]')
      AND  type = 'FN'
)
BEGIN
    DROP FUNCTION [dbo].[fnDateTimeToFileTime]
END
GO

-- Create function.
CREATE FUNCTION [dbo].[fnDateTimeToFileTime]
(
    @DateTime AS DATETIME
)
RETURNS
    BIGINT
BEGIN

IF @DateTime IS NULL
    RETURN NULL

DECLARE @MsecBetween1601And1970 BIGINT
DECLARE @MsecBetween1970AndDate BIGINT

SET @MsecBetween1601And1970 = 11644473600000

SET @MsecBetween1970AndDate = 
    DATEDIFF(ss, CAST('1970-01-01 00:00:00' as DATETIME), @DateTime) * 
        CAST(1000 AS BIGINT)

RETURN (@MsecBetween1601And1970 + @MsecBetween1970AndDate) * CAST(10000 AS BIGINT)  
END
GO

IF @@ERROR = 0
    GRANT EXECUTE ON [dbo].[fnDateTimeToFileTime] TO Public 
GO

Кажется, с точностью до 1 секунды, что нормально (я не могу сделать его более точным из-за переполнения данных) Я использовал веб-инструмент TimeAndDate для расчета продолжительности между датами.

Что ты думаешь?

2 голосов
/ 27 мая 2010

2 эра времени SQL Server начинается 1900-01-01 00:00:00 (за SELECT CAST (0 как дата).

Нет, это базовая дата, дата-время начинается с 1753

запустить это

select cast('17800122' as datetime) 

выход

1780-01-22 00: 00: 00.000

Но это все же меньше времени файла, поэтому вам нужно добавить это ... однако помните григорианский и юлианский календари (также причина, по которой дата и время начинается в 1753 году)

...