大家好
我正在尝试构建一个用于解析特定领域语言的词法分析器。我有一组保留令牌(片段 RESERVED)和一个转义字符。每当出现未转义的保留标记时,词法分析器都应该拆分。
一个简化的例子:
SEP: ';';
AND: '&&';
fragment ESCAPE: '/';
fragment RESERVED: SEP | AND | ESCAPE;
SINGLETOKEN : (~(RESERVED) | (ESCAPE RESERVED))+;
问题:
只要 RESERVED 仅包含单个字符标记,此方法就可以正常工作。否定操作 ~ 仅适用于单个字符。
不幸的是,我也需要它与字符串令牌一起使用。所以标记超过 1 个字符(参见示例中的 AND)。有没有一种简单的方法可以做到这一点?我需要在没有内联 java 或 c 代码的情况下解决问题,因为它必须编译成不同的语言,而且我不想维护单独的副本。
我希望有一个人可以帮助我
来自整个脚本的示例输入
创建;假;假;1.key = bla;trig;true;false;(1.key1 ~ .*thisIsRegex || 2.oldKey1 €) && (1.bla=2.blub || 1.blub=bla);
在 Lexer 之后,这应该看起来像这样 | 是令牌分隔符,空格不重要:|create|;|false|;|false|;|1.|key| = |白|;| 触发|;|true|;|false|;|(|1.|key1| ~| .*thisIsRegex| || |2.|oldKey1| €|)| && |(|1.|bla|=|2.|blub| || |1.|blub|=|bla|)|;|
整个脚本可以在http://pastebin.com/Cz520VW4上找到 (注意这个链接会在一个月内过期)它目前不适用于正则表达式部分。
可能但可怕的解决方案
我找到了一个可能的解决方案,但它真的很hacky并且使脚本更容易出错。所以我宁愿找更干净的东西。
我目前正在做的是手写否定(~RESERVED)。
SEP: ';';
AND: '&&';
fragment ESCAPE: '/';
fragment RESERVED: SEP | AND | ESCAPE;
NOT_RESERVED:
: '&' ~('&' | SEP | ESCAPE)
// any two chars starting with '&' followed by a character other then a reserve character
| ~('&' | SEP | ESCAPE) ~(SEP | ESCAPE)
// other than reserved character followed by '&' followed by any char
;
SINGELTON : (NOT_RESERVED | (ESCAPE RESERVED))+;
真正的脚本有超过 5 个多字符标记,以后可能会有更多超过 2 个字符,所以这种解决问题的方法会变得相当复杂。