Как перевести значение из базы в DateTime?

не могу понять в каком формате хранится время в таблице базы данных. Тип данных в таблице - bigint. Время в таблицу записывается с интервалом в один час. Одно из значений времени равно 637976910630000000. Предположил что время пишется в базу в Unix формате, но это очень огромное число, пробовал делать различные преобразования, но вменяемого результата не получил, получается очень большая дата. База данных MS SQL. В то же время есть софтина, которая тоже запрашивает данные из этой таблицы и рисует графики, там время выводится корректно. В запрос этой программы залезть возможности нет. Помогите преобразовать это значение в нормальный формат даты.


Ответы (3 шт):

Автор решения: MrDrBaton

Есть на англоязычном варианте стака. select dateadd(s, convert(bigint, 1283174502729) / 1000, convert(datetime, '1-1-1970 00:00:00')). Ищите по "Convert bigint to datetime"

→ Ссылка
Автор решения: Arkee

Скорее такая цифра получена из даты с точностью до 100 наносекунд. Допустим, вот такой пример:

DECLARE @num [bigint] = 637976910639900123;

Для парсинга даты я бы разбил это значение на несколько сегментов: дни для подсчета даты, секунды для подсчета времени и наносекунды чтобы не терять точность.

Начало даты не unixtimestamp. Значение слишком велико для этого. Скорее это 1 января 0001 года.

Находим делители дат:

DECLARE @nanoSecPrecise [bigint] = 100;

DECLARE @dayHours [bigint] = 24;
DECLARE @hourMinutes [bigint] = 60;
DECLARE @minuteSeconds [bigint] = 60;
DECLARE @secondMilliseconds [bigint] = 1000;
DECLARE @millisecondsMicroseconds [bigint] = 1000;
DECLARE @microsecondsNanoseconds [bigint] = 1000 / @nanoSecPrecise;

DECLARE @daySeparator [bigint] = @dayHours * @hourMinutes * @minuteSeconds * @secondMilliseconds * @millisecondsMicroseconds * @microsecondsNanoseconds;
DECLARE @secondsSeparator [bigint] = @secondMilliseconds * @millisecondsMicroseconds * @microsecondsNanoseconds;

И находим дату:

DECLARE @days [bigint] = @num / @daySeparator;
DECLARE @time [bigint] = (@num % @daySeparator) / @secondsSeparator;
DECLARE @nanoseconds [bigint] =  ((@num % @daySeparator) % @secondsSeparator) * @nanoSecPrecise;

DECLARE @startDate [datetime2] = CONVERT([datetime2], '0001-01-01 00:00:00.000', 121);
DECLARE @dayAdds [datetime2] = DATEADD([DAY], @days, @startDate);
DECLARE @timeAdds [datetime2] = DATEADD([SECOND], @time, @dayAdds);
DECLARE @dt2Precision [datetime2] = DATEADD([NANOSECOND], @nanoseconds, @timeAdds);

Результат можно проверить:

SELECT
  @daySeparator AS [@daySeparator],
  @secondsSeparator AS [@secondsSeparator],
  @days AS [@days],
  @time AS [@time],
  @nanoseconds AS [@nanoseconds],
  @startDate AS [@startDate],
  @dayAdds AS [@dayAdds],
  @timeAdds AS [@timeAdds],
  @dt2Precision AS [@dt2Precision]

Вывод:

@daySeparator @secondsSeparator @days @time @nanoseconds @startDate @dayAdds @timeAdds @dt2Precision
864000000000 10000000 738399 17463 990012300 0001-01-01 00:00:00.0000000 2022-09-02 00:00:00.0000000 2022-09-02 04:51:03.0000000 2022-09-02 04:51:03.9900123
→ Ссылка
Автор решения: Alexey Trukhanov

Я совсем не понимаю в SQLServer, сам использую postgresql, однако, стало интересно, что это за дата и я провел ряд экспериментов. Конечно, сильно бы помогло с поиском верного ответа, если бы автор хотя бы приблизительно назвал реальные даты событий, тогда было бы с чем сопоставлять. Если предположить, что это наносекунды в UNIXtime, то деление даты на 10^9 дает вполне адекватный результат, но, повторю, без понимания, что это за события и когда они в реальности произошли, не понятно, насколько всеже результат адекватен.

Эксперимент на питоне:

from datetime import datetime

datetime_start = str(datetime.fromtimestamp(637976910630000000 / 1_000_000_000))

print(datetime_start)

Дает результат:

1990-03-21 02:48:30.630000

Решение для SQLServer

Для конвертации этого числа в дату в SQLServer нашел решение через сложение секунд со стартовой датой:

Select
    dateadd(S, [unixtime], '1970-01-01')
From [Table]

Взято отсюда, сам не пробовал.

→ Ссылка