如何用localdatetime解析/格式化日期( java 8)

0s7z1bwu  于 2021-07-13  发布在  Java
关注(0)|答案(4)|浏览(190)

Java8为处理日期和时间添加了一个新的java.time api(JSR310)。
我把日期和时间作为字符串(例如。 "2014-04-08 12:30" ). 我怎样才能获得 LocalDateTime 给定字符串中的示例?
在我完成与 LocalDateTime 对象:然后如何转换 LocalDateTime 示例返回到与上面显示的格式相同的字符串?

erhoui1w

erhoui1w1#

分析日期和时间
创建 LocalDateTime 对象,可以使用静态 LocalDateTime.parse() 方法。它需要一根绳子和一根绳子 DateTimeFormatter 作为参数。这个 DateTimeFormatter 用于指定日期/时间模式。

String str = "1986-04-08 12:30";
DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm");
LocalDateTime dateTime = LocalDateTime.parse(str, formatter);

格式化日期和时间
创建格式化字符串的步骤 LocalDateTime 对象可以使用 format() 方法。

DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm");
LocalDateTime dateTime = LocalDateTime.of(1986, Month.APRIL, 8, 12, 30);
String formattedDateTime = dateTime.format(formatter); // "1986-04-08 12:30"

请注意,在中有一些常用的日期/时间格式预定义为常量 DateTimeFormatter . 例如:使用 DateTimeFormatter.ISO_DATE_TIME 格式化 LocalDateTime 上面的示例将生成字符串 "1986-04-08T12:30:00" .
这个 parse() 以及 format() 方法可用于所有与日期/时间相关的对象(例如。 LocalDate 或者 ZonedDateTime )

bvn4nwqk

bvn4nwqk2#

你也可以使用 LocalDate.parse() 或者 LocalDateTime.parse()String 如果 String 是iso-8601格式。
例如,

String strDate = "2015-08-04";
LocalDate aLD = LocalDate.parse(strDate);
System.out.println("Date: " + aLD);

String strDatewithTime = "2015-08-04T10:11:30";
LocalDateTime aLDT = LocalDateTime.parse(strDatewithTime);
System.out.println("Date with Time: " + aLDT);

输出,

Date: 2015-08-04
Date with Time: 2015-08-04T10:11:30

使用 DateTimeFormatter 只有当你必须处理其他的约会模式。
例如,在下面的示例中,dd mmm uuu表示月份的日期(两位数)、月份名称的三个字母(一月、二月、三月……)和四位数的年份:

DateTimeFormatter dTF = DateTimeFormatter.ofPattern("dd MMM uuuu");
String anotherDate = "04 Aug 2015";
LocalDate lds = LocalDate.parse(anotherDate, dTF);
System.out.println(anotherDate + " parses to " + lds);

输出

04 Aug 2015 parses to 2015-08-04

还要记住 DateTimeFormatter 对象是双向的;它既可以解析输入,也可以格式化输出。

String strDate = "2015-08-04";
LocalDate aLD = LocalDate.parse(strDate);
DateTimeFormatter dTF = DateTimeFormatter.ofPattern("dd MMM uuuu");
System.out.println(aLD + " formats as " + dTF.format(aLD));

输出

2015-08-04 formats as 04 Aug 2015

(请参阅格式化和解析dateformatter的模式的完整列表)

Symbol  Meaning                     Presentation      Examples
  ------  -------                     ------------      -------
   G       era                         text              AD; Anno Domini; A
   u       year                        year              2004; 04
   y       year-of-era                 year              2004; 04
   D       day-of-year                 number            189
   M/L     month-of-year               number/text       7; 07; Jul; July; J
   d       day-of-month                number            10

   Q/q     quarter-of-year             number/text       3; 03; Q3; 3rd quarter
   Y       week-based-year             year              1996; 96
   w       week-of-week-based-year     number            27
   W       week-of-month               number            4
   E       day-of-week                 text              Tue; Tuesday; T
   e/c     localized day-of-week       number/text       2; 02; Tue; Tuesday; T
   F       week-of-month               number            3

   a       am-pm-of-day                text              PM
   h       clock-hour-of-am-pm (1-12)  number            12
   K       hour-of-am-pm (0-11)        number            0
   k       clock-hour-of-am-pm (1-24)  number            0

   H       hour-of-day (0-23)          number            0
   m       minute-of-hour              number            30
   s       second-of-minute            number            55
   S       fraction-of-second          fraction          978
   A       milli-of-day                number            1234
   n       nano-of-second              number            987654321
   N       nano-of-day                 number            1234000000

   V       time-zone ID                zone-id           America/Los_Angeles; Z; -08:30
   z       time-zone name              zone-name         Pacific Standard Time; PST
   O       localized zone-offset       offset-O          GMT+8; GMT+08:00; UTC-08:00;
   X       zone-offset 'Z' for zero    offset-X          Z; -08; -0830; -08:30; -083015; -08:30:15;
   x       zone-offset                 offset-x          +0000; -08; -0830; -08:30; -083015; -08:30:15;
   Z       zone-offset                 offset-Z          +0000; -0800; -08:00;

   p       pad next                    pad modifier      1

   '       escape for text             delimiter
   ''      single quote                literal           '
   [       optional section start
   ]       optional section end
   #       reserved for future use
   {       reserved for future use
   }       reserved for future use
pengsaosao

pengsaosao3#

以上两个答案都很好地解释了关于弦模式的问题。但是,如果您正在使用iso 8601,则无需应用 DateTimeFormatter 因为localdatetime已经准备好了:
将localdatetime转换为时区iso8601字符串

LocalDateTime ldt = LocalDateTime.now(); 
ZonedDateTime zdt = ldt.atZone(ZoneOffset.UTC); //you might use a different zone
String iso8601 = zdt.toString();

从iso8601字符串转换回localdatetime

String iso8601 = "2016-02-14T18:32:04.150Z";
ZonedDateTime zdt = ZonedDateTime.parse(iso8601);
LocalDateTime ldt = zdt.toLocalDateTime();
sauutmhj

sauutmhj4#

将带有日期和时间的字符串解析为特定的时间点(java称之为 Instant )相当复杂。java已经在几次迭代中解决了这个问题。最新的一个, java.time 以及 java.time.chrono ,涵盖了几乎所有的需求(除了时间膨胀:)。
然而,这种复杂性带来了很多困惑。
理解日期解析的关键是:

为什么java有这么多方法来解析日期

有几种测量时间的系统。例如,历史上的日本历法是从各个皇帝或王朝的统治时期衍生出来的。然后是unix时间戳。幸运的是,整个(商业)世界都设法使用同样的方法。
历史上,由于各种原因,系统从/切换到。e、 从儒略历到1582年的公历。所以在此之前的“西方”约会需要区别对待。
当然,这种变化并没有立即发生。因为历法来源于某些宗教的总称,而欧洲其他地区信奉其他饮食,例如德国直到1700年才改用。

…为什么localdatetime,ZoneDateTime等这么复杂

有时区。时区基本上是地球表面的一条“条纹”*[1],其权威遵循相同的规则,即何时有时间偏移。这包括夏季规则。
不同地区的时区随着时间的推移而变化,主要取决于谁征服了谁。一个时区的规则也会随着时间的推移而改变。
有时间偏移。这与时区不同,因为时区可以是例如“布拉格”,但它有夏季时间偏移和冬季时间偏移。
如果你得到一个带有时区的时间戳,偏移量可能会有所不同,这取决于它在一年中的哪个部分。在闰时,时间戳可能意味着两个不同的时间,因此如果没有额外的信息,就无法可靠地转换。
注:timestamp的意思是“包含日期和/或时间的字符串,可以选择带有时区和/或时间偏移量。”
在某些时段内,多个时区可能共享同一时间偏移。例如,当夏季时间偏移无效时,gmt/utc时区与“伦敦”时区相同。
使其更复杂一些(但这对于您的用例来说并不太重要):
科学家们观察地球的动态,它随时间而变化;基于此,他们在每一年的末尾加上秒数(所以呢 2040-12-31 24:00:00 可能是有效的日期时间。)这需要定期更新系统用于正确进行日期转换的元数据。e、 g.在linux上,您可以定期更新java包,包括这些新数据。
对于历史和将来的时间戳,更新并不总是保持以前的行为。因此,在不同版本的软件上运行时,对某个时区更改前后的两个时间戳进行分析比较可能会得到不同的结果。这也适用于比较受影响的时区和其他时区。
如果这会导致软件出现错误,请考虑使用一些没有复杂规则的时间戳,如unix时间戳。
因为7,对于未来的日期,我们不能确切地转换日期。例如,当前对 8524-02-17 12:00:00 可能会在未来几秒钟内关闭。

jdk的api是随着当代的需求而发展的

早期的java版本刚刚发布 java.util.Date 这有点天真,假设只有年,月,日和时间。这很快就不够了。
而且,数据库的需求不同,所以很早, java.sql.Date 它的引入有其自身的局限性。
因为两者都没有涵盖不同的日历和时区 Calendar 引入了api。
这仍然没有涵盖时区的复杂性。然而,上述api的混合使用确实是一种痛苦。因此,随着java开发人员开始开发全球web应用程序,针对大多数用例的库(如jodatime)迅速流行起来。约达时间是事实上的标准大约十年。
但是jdk没有与jodatime集成,因此使用它有点麻烦。因此,在对如何处理这个问题进行了很长时间的讨论之后,jsr-310主要是基于jodatime创建的。

如何在java的java.time中处理它

确定分析时间戳的类型

当您使用时间戳字符串时,您需要知道它包含哪些信息。这是关键点。如果您没有正确理解这一点,您最终会遇到一个神秘的异常,如“无法创建即时”或“区域偏移丢失”或“未知区域id”等。
无法从temporalaccessor获取offsetdatetime
无法从temporalaccessor获取zoneddatetime
无法从temporalaccessor获取localdatetime
无法从临时加速器获取即时消息
它包含日期和时间吗?
它有时间偏移吗?
时间偏移是 +hh:mm 部分。有时候, +00:00 可替换为 Z 作为“祖鲁时间”, UTC 世界时协调,或 GMT 和格林威治标准时间一样。这些也设置了时区。
对于这些时间戳,使用 OffsetDateTime .
有时区吗?
对于这些时间戳,使用 ZonedDateTime .
区域由指定
名称(“布拉格”,“太平洋标准时间”,“太平洋标准时间”),或
“zone id”(“美国/洛杉矶”,“欧洲/伦敦”),由java.time.zoneid表示。
时区列表由icaan支持的“tz数据库”编制。
根据 ZoneId 的javadoc,区域id也可以指定为 Z 和偏移量。我不知道这是怎么Map到真实区域的。如果时间戳(只有一个tz)落入时间偏移量变化的闰时,那么它是不明确的,解释是有问题的 ResolverStyle ,见下文。
如果两者都没有,则假定或忽略缺失的上下文。消费者必须做出决定。所以它需要被解析为 LocalDateTime 并转化为 OffsetDateTime 通过添加缺少的信息:
您可以假设这是utc时间。添加0小时的utc偏移量。
你可以假设这是转换发生的时间。通过添加系统的时区来转换它。
你可以忽略它,就这样使用它。这很有用,例如比较或减去两次(参见 Duration ),或者当你不知道并且这并不重要时(例如,当地的公共汽车时刻表)。
部分时间信息
根据时间戳包含的内容,您可以 LocalDate , LocalTime , OffsetTime , MonthDay , Year ,或 YearMonth 从中解脱出来。
如果你有完整的信息,你可以得到一个 java.time.Instant . 这也是内部用于在 OffsetDateTime 以及 ZonedDateTime .

找出如何解析它

有大量关于 DateTimeFormatter 它既可以解析时间戳字符串,也可以格式化为字符串。
预先创建的 DateTimeFormatter s应该涵盖更多的标准时间戳格式。例如, ISO_INSTANT 可以解析 2011-12-03T10:15:30.123457Z .
如果您有一些特殊的格式,那么您可以创建自己的datetimeformatter(它也是一个解析器)。

private static final DateTimeFormatter TIMESTAMP_PARSER = new DateTimeFormatterBuilder()
   .parseCaseInsensitive()
   .append(DateTimeFormatter.ofPattern("yyyy-MM-dd'T'HH:mm:ss.SX"))
   .toFormatter();

我建议看一下 DateTimeFormatter 并获得如何使用 DateTimeFormatterBuilder . 当你在那里的时候,也看看 ResolverStyle 它控制解析器对格式和模糊信息是宽松的、智能的还是严格的。

临时接受器

现在,经常犯的错误是深入到 TemporalAccessor . 这源于开发人员习惯于与 SimpleDateFormatter.parse(String) . 正确的, DateTimeFormatter.parse("...") 给你 TemporalAccessor .

// No need for this!
TemporalAccessor ta = TIMESTAMP_PARSER.parse("2011-... etc");

但是,利用上一节的知识,您可以方便地解析为所需的类型:

OffsetDateTime myTimestamp = OffsetDateTime.parse("2011-12-03T10:15:30.123457Z", TIMESTAMP_PARSER);

你实际上不需要 DateTimeFormatter 任何一个。要分析的类型具有 parse(String) 方法。

OffsetDateTime myTimestamp = OffsetDateTime.parse("2011-12-03T10:15:30.123457Z");

关于 TemporalAccessor ,您可以使用

相关问题