我有一个这种格式的字符串:"2019-08-17T09:51:41.775+00:00"。我需要将其转换为大纪元微秒,但我的转换总是一个小时。
这是我目前的代码:
String timestamp = "2019-08-17T09:51:41.775+00:00"
ZonedDateTime date = ZonedDateTime.parse(timestamp)
Long epoch = date.toInstant().toEpochMilli() * 1000
因此,当我运行此代码时,我得到的结果1566035501775000。现在,如果我1566035501775放入像这样的转换器中:https://www.freeformatter.com/epoch-timestamp-to-date-converter.html 颠倒它,我会得到这个日期:8/17/2019,10:51:41 AM。
为什么休息一个小时,我该如何改变?
我还尝试了使用DateTime格式化程序:
DateTimeFormatter pat = DateTimeFormatter.ofPattern("YYYY-MM-dd'T'hh:mm:ss.SSSXXX")
Instant.from(pat.parse(timestamp))
但这给了我这个例外:
java.time.DateTimeException: Unable to obtain Instant from TemporalAccessor: {SecondOfMinute=41, MicroOfSecond=775000, DayOfMonth=17, HourOfAmPm=9, NanoOfSecond=775000000, MonthOfYear=8, OffsetSeconds=0, MinuteOfHour=51, MilliOfSecond=775, WeekBasedYear[WeekFields[SUNDAY,1]]=2019},ISO of type java.time.format.Parsed
at java.time.Instant.from(Instant.java:378)
at com.amazon.hailstonetracing.controller.chrome.WorkflowEventMapperTest.test(WorkflowEventMapperTest.kt:114)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:498)
at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47)
at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17)
at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:26)
at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:78)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:57)
at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290)
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58)
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268)
at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
at org.junit.runner.JUnitCore.run(JUnitCore.java:137)
at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:68)
at com.intellij.rt.execution.junit.IdeaTestRunner$Repeater.startRunnerWithArgs(IdeaTestRunner.java:47)
at com.intellij.rt.execution.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:242)
at com.intellij.rt.execution.junit.JUnitStarter.main(JUnitStarter.java:70)
Caused by: java.time.temporal.UnsupportedTemporalTypeException: Unsupported field: InstantSeconds
at java.time.format.Parsed.getLong(Parsed.java:203)
at java.time.Instant.from(Instant.java:373)
... 24 more
经过研究,我认为这是因为我没有定义 ZoneId?我只有时间戳字符串的偏移量。但是如果我在DateTimeFormatter
上指定.withZone(ZoneId.of("UTC"))
,我会得到相同的结果.
感谢您的帮助!
您的微秒值是正确的。
发生的情况是:freeformatter.com Epoch和Unix时间戳转换器会检测您的时区,并为您提供本地时间,上午10:51:41。显然,您(或您的浏览器认为您(处于与 UTC 偏移 +01:00 的时区。因此,与字符串中指定为偏移量 +00:00 的时间相比,它增加了一个小时。
我在欧洲/哥本哈根时区,目前偏移量 +02:00,当我在 freeformatter.com 上尝试相同的操作时,我得到 17.8.2019 11.51.41,这是我时区的正确当地时间。
在尝试使用显式格式化程序时,您有一些格式模式字母的错误大小写。由于它无需构建自己的格式化程序即可工作,因此这当然是我推荐的解决方案。
而且您不需要时区 ID。字符串中的偏移量是完全足够的。
由于您的字符串有偏移量(+00:00(并且没有时区(例如太平洋/塔拉瓦(,因此使用ZonedDateTime
是矫枉过正的。我建议改用OffsetDateTime
。它以相同的方式进行:
OffsetDateTime date = OffsetDateTime.parse(timestamp);
long epoch = date.toInstant().toEpochMilli() * 1000;
我还将epoch
的类型从Long
对象更改为原始long
。
顺便说一句,如果您的字符串具有比毫秒更精细的精度,那么它会在您的转换中丢失。以下是获得微秒级精度的方法:
String timestamp = "2019-08-17T09:51:41.7754321+00:00";
OffsetDateTime date = OffsetDateTime.parse(timestamp);
Instant asInstant = date.toInstant();
long epoch = TimeUnit.SECONDS.toMicros(asInstant.getEpochSecond())
+ asInstant.get(ChronoField.MICRO_OF_SECOND);
System.out.println(epoch);
如您所见,我在您的字符串中添加了更多小数用于演示。输出为:
1566035501775432