将字符串转换为即时tl;博士使用个位数格式模式仅使用 java.time 关于 java.time

我正在尝试使用java 8或utils包在字符串中隐蔽日期时间到即时。

例如。

String requestTime = "04:30 PM, Sat 5/12/2018";

Instant reqInstant should result in 2018-05-12T20:30:00.000Z

reqString位于美国/多伦多时区。

这就是我尝试过的

 String strReqDelTime = "04:30 PM, Sat 5/12/2018";
 Date date = new SimpleDateFormat("hh:mm a, EEE MM/dd/yyyy").parse(requestTime);
 Instant reqInstant = date.toInstant();

上述代码的结果是 ."2018-05-12T23:30:00Z"

任何帮助是值得赞赏的。


答案 1

tl;博士

  • 修复未填充的月份和日期的格式模式。
  • 仅使用 java.time 类,而不使用旧版类。

人为示例:

LocalDateTime.parse(                   // Parse as an indeterminate `LocalDate`, devoid of time zone or offset-from-UTC. NOT a moment, NOT a point on the timeline.
    "04:30 PM, Sat 5/12/2018" ,        // This input uses a poor choice of format. Whenever possible, use standard ISO 8601 formats when exchanging date-time values as text. Conveniently, the java.time classes use the standard formats by default when parsing/generating strings.
    DateTimeFormatter.ofPattern( "hh:mm a, EEE M/d/uuuu" , Locale.US )  // Use single-character `M` & `d` when the number lacks a leading padded zero for single-digit values.
)                                      // Returns a `LocalDateTime` object.
.atZone(                               // Apply a zone to that unzoned `LocalDateTime`, giving it meaning, determining a point on the timeline.
    ZoneId.of( "America/Toronto" )     // Always specify a proper time zone with `Contintent/Region` format, never a 3-4 letter pseudo-zone such as `PST`, `CST`, or `IST`. 
)                                      // Returns a `ZonedDateTime`. `toString` → 2018-05-12T16:30-04:00[America/Toronto].
.toInstant()                           // Extract a `Instant` object, always in UTC by definition.
.toString()                            // Generate a String in standard ISO 8601 format representing the value within this `Instant` object. Note that this string is *generated*, not *contained*.

2018-05-19T10:43:00Z

使用个位数格式模式

您在格式设置模式中使用了表示任何个位数值(1 月至 9 月)都将以带衬垫的前导零显示。MM

但是您的输入缺少填充的前导零。因此,请使用单个.M

同上,我期望的月份中的某一天:而不是.ddd

仅使用 java.time

您正在使用麻烦的有缺陷的旧日期时间类( &),这些类在几年前被java.time类所取代。新类完全取代了旧类。无需将传统与现代混为一谈。DateSimpleDateFormat

LocalDateTime

解析为 LocalDateTime,因为输入字符串缺少时区UTC 偏移量的任何指示器。这样的值不是一瞬间,不是时间轴上的一个点。它只是沿着大约26-27小时范围的一组潜在时刻。

String input = "04:30 PM, Sat 5/12/2018";
DateTimeFormatter f = DateTimeFormatter.ofPattern( "hh:mm a, EEE M/d/uuuu" , Locale.US );  // Specify locale to determine human language and cultural norms used in translating that input string.
LocalDateTime ldt = LocalDateTime.parse( input , f );

北京时间:2018-05-12T16:30

ZonedDateTime

如果您确定输入旨在使用加拿大多伦多地区的人们使用的挂钟时间来表示时刻,请应用 a 以获取 ZonedDateTime 对象。ZoneId

分配时区为未分区赋予了意义。现在我们有一个时刻,时间轴上的一个点。LocalDateTime

ZoneId z = ZoneId.of( "America/Toronto" ) ;
ZonedDateTime zdt = ldt.atZone( z ) ;  // Give meaning to that `LocalDateTime` by assigning the context of a particular time zone. Now we have a moment, a point on the timeline.

zdt.toString(): 2018-05-12T16:30-04:00[美国/多伦多]

Instant

要查看与 UTC 相同的时刻,请提取即时。相同的时刻,不同的挂钟时间。

Instant instant = zdt.toInstant() ;

瞬换弦(): 2018-05-12T20:30:00Z


关于 java.time

java.time 框架内置于 Java 8 及更高版本中。这些类取代了麻烦的旧日期时间类,如java.util.DateCalendarSimpleDateFormat

Joda-Time 项目现在处于维护模式,建议迁移到 java.time 类。

要了解更多信息,请参阅 Oracle 教程。搜索 Stack Overflow 以获取许多示例和解释。规格是JSR 310

您可以直接与数据库交换 java.time 对象。使用符合 JDBC 4.2 或更高版本的 JDBC 驱动程序。不需要字符串,不需要类。java.sql.*

从哪里获取 java.time 类?

ThreeTen-Extra 项目通过其他类扩展了 java.time。这个项目是未来可能添加到java.time的试验场。您可以在此处找到一些有用的课程,例如IntervalYearWeekYearQuarter


答案 2

您的计算机(服务器)中的时区似乎是美国太平洋夏令时(GMT-7),但您希望美国东部夏令时(GMT-4)的结果。

Instant.toString() 以 ISO-8601 格式返回 UTC (GMT+0) DateTime。(末尾的“Z”表示 UTC)。

简单日期格式在未指定时,在计算机的默认时区中处理日期时间字符串。并且您的输入不会指定时区。

因此,您需要对输入所在的时区执行一些操作。

PS.在东部DST的矿床上,您的代码给我的结果完全符合您的预期。