gpt4 book ai didi

java - 在开头创建一个带有可选部分的 DateTimeFormater

转载 作者:行者123 更新时间:2023-12-04 02:25:34 25 4
gpt4 key购买 nike

我有这种结构的时间码 hh:mm:ss.SSS 我有一个自己的类,实现时间接口(interface)。它具有自定义字段 TimecodeHour 字段,允许小时值大于 23。我想用 DateTimeFormatter 解析。小时值可选(可以省略,小时可以大于24);作为正则表达式 (\d*\d\d:)?\d\d:\d\d.\d\d\d

出于此问题的目的,我的自定义字段可以替换为正常的 HOUR_OF_DAY 字段。

我当前的格式化程序

DateTimeFormatter UNLIMITED_HOURS = new DateTimeFormatterBuilder()
.appendValue(ChronoField.HOUR_OF_DAY, 2, 2,SignStyle.NEVER)
.appendLiteral(':')
.parseDefaulting(TimecodeHour.HOUR, 0)
.toFormatter(Locale.ENGLISH);
DateTimeFormatter TIMECODE = new DateTimeFormatterBuilder()
.appendOptional(UNLIMITED_HOURS)
.appendValue(MINUTE_OF_HOUR, 2)
.appendLiteral(':')
.appendValue(SECOND_OF_MINUTE, 2)
.appendFraction(MILLI_OF_SECOND, 3, 3, true)
.toFormatter(Locale.ENGLISH);

具有小时值的时间码按预期解析,但具有小时值的值会抛出异常

java.time.format.DateTimeParseException: Text '20:33.123' could not be parsed at index 5

我假设,由于小时和分钟具有相同的模式,解析器从前面开始并捕获可选部分的分钟值。这样对吗,怎么解决?

最佳答案

我开始怀疑 20:33.123 并不是要表示一天中午夜过后 20 到 21 分钟之间的时间。也许相当多的时间,比 20 分钟长一点。如果这是正确的,请为其使用 Duration

不幸的是,java.time 不包括以 ISO 8601 格式以外的格式解析和格式化 Duration 的方法。这至少给我们留下了三个选择:

  1. 使用第三方库。 Time4J 提供了一个优雅的解决方案,见下文。 Joda-Time 有它的 PeriodFormatter 类。 Apache 还可能提供用于解析和格式化持续时间的工具。
  2. 在使用 Duration.parse() 解析之前将您的字符串转换为 ISO 8601 格式。
  3. 编写您自己的解析器。

我在想我们对 3. 太懒了,而且 Joda-Time 已经过时了,所以我想在这里追求选项 1. 和 2.,选项 1. 在 Time4J 变体中。

适配 ISO 8601 的正则表达式

ISO 8601 格式起初感觉很不寻常,但很简单。 PT20M33.123S 表示 20 分 33.123 秒。

public static Duration parse(String timeCodeString) {
String iso8601 = timeCodeString
.replaceFirst("^(\\d{2,}):(\\d{2}):(\\d{2}\\.\\d{3})$", "PT$1H$2M$3S")
.replaceFirst("^(\\d{2}):(\\d{2}\\.\\d{3})$", "PT$1M$2S");
return Duration.parse(iso8601);
}

让我们试试看:

    System.out.println(parse("20:33.123"));
System.out.println(parse("123:20:33.123"));

输出是:

PT20M33.123S
PT123H20M33.123S

我对 replaceFirst 的两次调用首先处理了小时数的情况,然后是没有小时数的情况。因此,两者都会将与您的正则表达式匹配的字符串转换为 ISO 8601 格式。 Duration 类随后解析。如您所见,Duration 还打印回 ISO 8601 格式。不过,以不同的方式格式化也不错,搜索如何。

时间4J

Time4J 库提供了非常优雅的解决方案,与您的思路非常相似。我们真正需要的是这个格式化程序:

private static final Formatter<ClockUnit> TIME_CODE_PARSER 
= Duration.formatter(ClockUnit.class, "[###hh:mm:ss.fff][mm:ss.fff]");

只需像这样使用:

    System.out.println(TIME_CODE_PARSER.parse("20:33.123"));
System.out.println(TIME_CODE_PARSER.parse("123:20:33.123"));
PT20M33,123000000S
PT123H20M33,123000000S

Time4J Duration 类也打印 ISO 8601 格式。似乎它使用逗号作为 ISO 8601 中首选的小数分隔符,并且当其中一些为 0 时,它也会在秒上打印 9 个小数。

格式模式字符串中###hh表示2到5位小时,fff表示秒的小数点后三位。

您的方法有什么问题吗?

你的方法有什么问题吗? ChronoField.HOUR_OF_DAY 表示:一天中的小时。 0 是午夜,12 是中午,23 是接近一天结束的时候。这不是您想要的,所以是的,您使用了错误的方法。虽然您可能可以让它正常工作,但在您之后维护您的代码的任何人都会发现它令人困惑,并且可能很难根据您的意图进行修改。

链接

关于java - 在开头创建一个带有可选部分的 DateTimeFormater,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/67935444/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com