Instant i1 = Instant.now();
Instant i2 = Instant.now().plusSeconds(60 * 60 * 24 * 365);
Duration<IsoUnit> dur = Duration.from((TemporalAmount) java.time.Duration.between(i1, i2));
System.out.println(dur);这段代码打印P365D,有没有办法让单元填满下一个更大的单元,这样就变成了P1Y,如果我有P334D到P11M之类的东西的话?
time4j版本为4.38
发布于 2019-05-22 05:19:18
其他评论和回答完全正确地说,一年并不总是等于365天。
Time4J-方法net.time4j.Duration.from(TemporalAmount)使用STD_PERIOD作为规范化器返回规范化持续时间。这个规范化器的文档说:
根据1年=12个月和1天= 24小时和1小时= 60分钟1分钟=60秒的基准将持续时间项目正常化,但不将天数转换为月份。
因此,您只能期望在以秒为单位定义的时间数量开始的几天内得到结果。
如果您还想要一年,那么我建议您首先使用适当的时区将即时日期转换为日历日期。但是在闰年,您的代码仍然会产生365天,而不是在将60 * 60 * 24 * 365秒添加到第二个瞬间之后一年。所以你增加的秒也是有缺陷的,因为它是基于错误的假设。
附带说明:
如果您希望采用相反的方式,即一年中有多少秒,那么您可以使用以下代码
Moment m1 = Moment.nowInSystemTime();
Moment m2 = m1.toZonalTimestamp(ZonalOffset.UTC).plus(1, CalendarUnit.YEARS).atUTC();
long seconds = SI.SECONDS.between(m1, m2); // = 366 days in seconds if applied on date 2019-05-22!随着未来版本的Time4J和可能的闰秒在2019年年底,代码甚至可能产生额外的秒。
无论如何,我建议您将Time4J更新为v5.4,并考虑以下映射:
java.time.Instant as input => net.time4j.MachineTime.from(...)
java.time.LocalDateTime/net.time4j.PlainTimestamp => net.time4j.Duration.from(...)因此,如果您确实希望在打印持续时间中尽可能输出年份,并且您有实例/矩,那么在创建适当的工期对象之前,首先将其转换为LocalDateTime/PlainTimestamp (使用时区或偏移量)。
2019年至05-25年的最新情况:
使用版本v5.4 (或更高版本)的另一种方法可以通过“模糊”持续时间来实现。您可以通过应用近似来规范所获得的持续时间。示例:
Duration<IsoUnit> d = Duration.of(60 * 60 * 24 * 365, ClockUnit.SECONDS);
d = d.with(Duration.approximateMaxUnitOnly());
System.out.println(d); // P1Y由于这种规范化的性质,您不能期望得到确切的结果。
发布于 2019-05-21 15:31:03
一年没有固定的秒数:
由于以上原因,你必须知道一年中有多少秒。与Instant和Duration不同,您应该使用LocalDate和Period (至少在使用java.time时):
LocalDate d1 = LocalDate.now();
LocalDate d2 = d1.plusYears(1);
Period p = Period.between(d1, d2);
System.out.println(p); // P1Yhttps://stackoverflow.com/questions/56240919
复制相似问题