How to “skip” a rule in order to favor one rule over the other?
我目前正在使用 antlr 构建日期解析器。它需要的输入是
1
2 |
year monthName numDayOfMonth era
numDayOfMonth monthName year era |
这些都在stringDate规则下,所以我的语法是这样的
1
2 3 4 5 6 7 8 9 |
stringDate: year monthName numDayOfMonth
| numDayOfMonth monthName year; numYear: NUMBER ; NUMBER: [0–9]+ ; |
在我的听众中,我检查以确保 numDayOfMonth 在 [1, 31] 范围内,以确保该数字是有效日期。我对几个月做同样的事情(首先我将它们转换成相应的月份)。
问题是,如果输入日期2013 June 13,日期会被正确解析。但是,当我输入 13 June 2013 时,它会被错误地解析,因为解析器会感到困惑并认为 2013 年是一天,而不是一年,因此在 exitNumDayOfMonth 期间检查失败。我一直在为如何处理这个问题而摸不着头脑。我本质上希望评估者跳过我遇到 num > 31 的规则,但我不完全确定如何跳过规则。我试过 return 并抛出错误,但似乎没有任何效果。
有没有办法让评估者跳过这条规则,转而使用替代方法?
- antlr 没有日期类吗?无论如何,如果您的年份总是 4 位数,请解析字符串并首先找到它。然后找到非数字月份并提取它。然后你就剩下这一天了。你也可以检查堆栈溢出。在 antlr 中已经有一些解析日期的线程。这是一个:stackoverflow.com/questions/35651518/…
为什么不将年份的标记定义更改为仅包含 4 位数字?这将解决问题。
所以,你的年份和日期将是
1
2 |
numYear: [0–9] [0–9] [0–9] [0–9]
numDayOfMonth: [0–9] | [0–9] [0–9] |
目前,它们都具有相同的定义 – 因此解析器在解析时不知道要选择哪个规则,而是选择第一个适合输入的规则。
来源:https://www.codenong.com/52845491/