5

我知道以毫秒为单位的 Unix 时间戳转换为我可以使用的 SQL 时间戳

SELECT TO_DATE('1970-01-01','YYYY-MM-DD HH24:MI:SS') + 
       (:timestamp / (1000*60*60*24)) FROM DUAL;

但我需要一个时间戳,所以我尝试了

SELECT TO_TIMESTAMP('1970-01-01 00:00:00','YYYY-MM-DD HH24:MI:SSFF3') + 
       (:timestamp) from DUAL

这给了我错误:

错误:ORA-01841:(完整)年份必须介于 -4713 和 +9999 之间,并且不能为 0

似乎在时间戳上加 1 总是会将其转换为一天。

我怎样才能得到一个真正的时间戳?

4

4 回答 4

6

timestamp如果将 an 添加interval到 a 中,您将得到a timestamp(请参阅日期/间隔算术)。

正如Benoit 注意到的那样,当它们超过大约 2.1e9 时,您不能指定以秒为单位的间隔:

SQL> SELECT numtodsinterval(2.2e9, 'SECOND'),
  2         numtodsinterval(2.3e9, 'SECOND')
  3    FROM dual;

NUMTODSINTERVAL(2.2E9,'SECOND'  NUMTODSINTERVAL(2.3E9,'SECOND'
------------------------------- -------------------------------
+000024855 03:14:07.147483647   +000024855 03:14:07.147483647

这就是为什么您应该使用不会丢失精度的分钟数。例如,假设:TS是unix时间戳(即一个数字):

SQL> variable ts number;
SQL> -- determining unix timestamp with nanosecond precision
SQL> BEGIN
  2     :ts := (to_date('2099-01-01 01:02:03', 'yyyy-mm-dd hh24:mi:ss')
  3              - date '1970-01-01') * 1000*60*60*24
  4            + 123.456789;
  5  END;
  6  /

ts
---------
4070912523123,456789

SQL> select timestamp '1970-01-01 00:00:00'
  2         + numtodsinterval((:ts)/1000/60, 'MINUTE')
  3    from dual;

TIMESTAMP'1970-01-0100:00:00'+NUMTODSINTERVAL((:TS)/1000/60,'MINUTE')
---------------------------------------------------------------------------
2099-01-01 01:02:03.123456789
于 2013-03-20T10:13:02.500 回答
2

有两种类型:

  • 时间戳
  • 间隔

间隔是减去时间戳得到的,把时间戳加在一起是没有意义的。

如果您需要获得毫秒间隔,我建议使用第二个间隔并将其除以 1000:

我可以建议:

SELECT timestamp'1970-01-01 00:00:00' + (interval '1888' second(9) / 1000)
  FROM dual

这里的问题是你不能在同一个时间戳文字中使用超过 9 个数字。

如果您需要向纪元添加 2,061,464,797,255 毫秒,我可以建议:

SELECT TIMESTAMP'1970-01-01 00:00:00'
       + INTERVAL '2' SECOND(9) * 1000000000
       + INTERVAL '061464797' SECOND(9)
       + INTERVAL '255' SECOND(3) / 1000
  FROM dual

你得到 2035-04-29 13:06:37.255000000

它似乎受到 2038 错误的影响:TIMESTAMP'1970-01-01 00:00:00' + 30 亿秒不起作用,而它适用于 20 亿秒。

于 2013-03-20T10:13:56.190 回答
1

我在这里发布了一些将纳秒转换为时间戳和时间戳转换为纳秒的方法。这些方法不受时区影响,具有纳秒级精度。

您只需要调整它以使用毫秒而不是纳秒。

SELECT TIMESTAMP '1970-01-01 00:00:00 UTC' + numtodsinterval(
    1598434427263 --Replace line with desired milliseconds
/ 1000, 'SECOND') AS TIMESTAMP FROM dual;

TIMESTAMP
26/08/20 09:33:47,263000000 UTC
于 2020-08-26T13:06:51.537 回答
0

采用

SELECT TIMESTAMP '1970-01-01 00:00:00.1234' + INTERVAL '1 00:00:00' DAY TO SECOND  
       AS ts
  FROM dual;
于 2013-03-20T10:03:18.800 回答