- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
为可以处理以下格式之一的输入的 LocalDate 实例提供灵活的解析器:
下面的类尝试处理第一个和第二个模式。解析适用于年份输入,但年份 + 月份会导致下面列出的异常。
import java.time.LocalDate;
import java.time.format.DateTimeFormatter;
import java.time.format.DateTimeFormatterBuilder;
import java.time.temporal.ChronoField;
public class DateTest {
public static void main(String[] args) {
DateTimeFormatter parser = new DateTimeFormatterBuilder()
.parseDefaulting(ChronoField.MONTH_OF_YEAR, 1)
.parseDefaulting(ChronoField.DAY_OF_MONTH, 1)
.appendPattern("yyyy")
.optionalStart().appendPattern("MM").optionalEnd().toFormatter();
System.out.println(parser.parse("2014", LocalDate::from)); // Works
System.out.println(parser.parse("201411", LocalDate::from)); // Fails
}
}
第二次 parse() 尝试导致以下异常:
Exception in thread "main" java.time.format.DateTimeParseException: Text '201411' could not be parsed at index 0
at java.time.format.DateTimeFormatter.parseResolved0(DateTimeFormatter.java:1949)
at java.time.format.DateTimeFormatter.parse(DateTimeFormatter.java:1851)
我认为我对可选部分模式的工作原理缺乏了解。我的目标是实现一个具有灵活格式的解析器,还是我需要检查输入长度并从解析器列表中进行选择?一如既往,我们将不胜感激。
最佳答案
问题的真正原因是符号处理。您的输入没有符号,但解析器元素“yyyy”贪婪地解析尽可能多的数字并期望一个正号,因为找到的数字超过四位。
我的分析以两种不同的方式进行:
调试(以查看不清楚的错误消息背后的真正原因)
基于我的 lib Time4J 在另一个解析引擎中模拟行为以获得更好的错误消息:
ChronoFormatter<LocalDate> cf =
ChronoFormatter
.ofPattern(
"yyyy[MM]",
PatternType.THREETEN,
Locale.ROOT,
PlainDate.axis(TemporalType.LOCAL_DATE)
)
.withDefault(PlainDate.MONTH_AS_NUMBER, 1)
.withDefault(PlainDate.DAY_OF_MONTH, 1)
.with(Leniency.STRICT);
System.out.println(cf.parse("201411"));
// java.text.ParseException: Positive sign must be present for big number.
您可以通过指示构建器始终仅使用四位数字来规避该问题:
DateTimeFormatter parser =
new DateTimeFormatterBuilder()
.appendValue(ChronoField.YEAR, 4)
.optionalStart()
.appendPattern("MM[dd]")
.optionalEnd()
.parseDefaulting(ChronoField.MONTH_OF_YEAR, 1)
.parseDefaulting(ChronoField.DAY_OF_MONTH, 1)
.toFormatter();
System.out.println(parser.parse("2014", LocalDate::from)); // 2014-01-01
System.out.println(parser.parse("201411", LocalDate::from)); // 2014-11-01
System.out.println(parser.parse("20141130", LocalDate::from)); // 2014-11-30
注意构建器中默认元素的位置。它们不是在开始时调用,而是在结束时调用,因为不幸的是,默认元素的处理在 java.time
中是位置敏感的。我还在第一个可选部分中为月份的日期添加了一个额外的可选部分。这个解决方案对我来说似乎更干净,而不是使用 Danila Zharenkov 建议的一系列 3 个可选部分,因为后者也可以解析具有更多数字的完全不同的输入(可能滥用可选部分作为 or-patterns 的替代品,特别是在宽松的情况下)解析)。
关于默认元素的位置敏感行为,这里引用自 API-documentation :
During parsing, the current state of the parse is inspected. If the specified field has no associated value, because it has not been parsed successfully at that point, then the specified value is injected into the parse result. Injection is immediate, thus the field-value pair will be visible to any subsequent elements in the formatter. As such, this method is normally called at the end of the builder.
顺便说一句:在我的 Time4J 库中,我还可以使用符号“|”定义真实或模式然后创建这个格式化程序:
ChronoFormatter<LocalDate> cf =
ChronoFormatter
.ofPattern(
"yyyyMMdd|yyyyMM|yyyy",
PatternType.CLDR,
Locale.ROOT,
PlainDate.axis(TemporalType.LOCAL_DATE)
)
.withDefault(PlainDate.MONTH_AS_NUMBER, 1)
.withDefault(PlainDate.DAY_OF_MONTH, 1)
.with(Leniency.STRICT);
关于具有可选模式的 Java DateTimeFormatterBuilder 导致 DateTimeParseException,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/50023654/
我已经配置了格式化程序: public static final DateTimeFormatter DATE_FORMATTER = new DateTimeFormatterBuilder()
这个问题已经有答案了: Parsing LocalDate in Java: What's the pattern for "March 26 2020" (2 个回答) 已关闭 2 年前。 我有一个
假设我有一个日期,例如: Nov 30, 2013 19:00:00.001930000 Eastern Standard Time 我正在尝试使用 DateTimeFormatterBuilder
我正在使用 DateTimeFormatterBuilder() 来转换我接收的 JSON 数据(注释代码)并将它们转换为两种格式之一。为了决定使用哪种格式,我使用 REGEX 来查找正方形、 []
我正在尝试将以下时间戳解析为 ZonedDateTime: Sun Jan 20 16:08:59 +0000 2019 我喜欢尽可能避免使用在字符串中定义的模式,因为我经常忘记字符代表什么(“哈,'
我正在尝试创建一个 DateTimeFormatter以匹配以下示例(它实际上比这稍微复杂一些,但这无关紧要)。 20180302-17:45:21 我写了以下内容,但它导致了异常: new Date
我需要解析各种格式的日期。其中一些缺少“日”。现在我想默认到月底。我不确定是否有直接的方法可以默认为月底 DateTimeFormatter f = new DateTimeFormatterBuil
先写代码。 @Test public void testDateFormat() { // this is ok DateTimeFormat.forPattern("yyyy-MM-
目标 为可以处理以下格式之一的输入的 LocalDate 实例提供灵活的解析器: 年 年月 年月日 实现尝试 下面的类尝试处理第一个和第二个模式。解析适用于年份输入,但年份 + 月份会导致下面列出的异
我的要求是根据指定的一组有效格式验证日期字符串的格式是否正确。 有效格式: MM/dd/yy MM/dd/yyyy 我创建了一个简单的测试方法,它使用 Java 8 DateTimeFormatter
我正在尝试解析从 Bitbucket event payload 收到的以下日期字符串: 2017-09-19T10:39:36+1000 当传入日期位于偏移量 +0000 中时,它可以工作,但 +1
我正在尝试从 Joda 迁移到 Java 8 的 ZonedDateTime,但我遇到了似乎无法解决的 DateTimeFormatterBuilder 问题。 我想接受以下任何一种格式: 2013-
我有以下格式化程序: DateTimeFormatter formatter = new DateTimeFormatterBuilder() .appendPattern("yyyy
我正在尝试使用 返回的 DateTimeFormatter 来解析表示具有 IsoField.QUARTER_OF_YEAR 的 LocalDate 值的文本DateTimeFormatterBuil
我目前有一个 Joda 日期解析器,它使用 DateTimeFormatterBuilder 以及我可能收到的六种不同日期格式。 我正在迁移到 Java 8 的 Date 例程,但没有看到等效的例程。
这个问题已经有答案了: Can’t rid of 'T' in LocalDateTime (3 个回答) 已关闭 3 年前。 我有这段代码可以将字符串双向解析为 LocalDateTime: pub
我是一名优秀的程序员,十分优秀!