AFAIK,错误是因为nestingBlockCommentCharacters
可以匹配+/
(~'/'
两次)。
就个人而言,我会保留nestingBlockComment
作为词法分析器规则而不是解析器规则。您可以通过在 lexer 类中添加一个小辅助方法来做到这一点:
public boolean openOrCloseCommentAhead() {
// return true iff '/+' or '+/' is ahead in the character stream
}
然后在词法分析器注释规则中,使用带有该辅助方法的门控语义谓词作为谓词内的布尔表达式:
// match nested comments
Comment
: '/+' (Comment | {!openOrCloseCommentAhead()}?=> Any)* '+/'
;
// match any character
Any
: .
;
一个小演示语法:
grammar DComments;
@lexer::members {
public boolean openOrCloseCommentAhead() {
return (input.LA(1) == '+' && input.LA(2) == '/') ||
(input.LA(1) == '/' && input.LA(2) == '+');
}
}
parse
: token+ EOF
;
token
: Comment {System.out.println("comment :: "+$Comment.text);}
| Any
;
Comment
: '/+' (Comment | {!openOrCloseCommentAhead()}?=> Any)* '+/'
;
Any
: .
;
和一个主要的类来测试它:
import org.antlr.runtime.*;
public class Main {
public static void main(String[] args) throws Exception {
ANTLRStringStream in = new ANTLRStringStream(
"foo /+ comment /+ and +/ comment +/ bar /+ comment +/ baz");
DCommentsLexer lexer = new DCommentsLexer(in);
CommonTokenStream tokens = new CommonTokenStream(lexer);
DCommentsParser parser = new DCommentsParser(tokens);
parser.parse();
}
}
然后是以下命令:
java -cp antlr-3.2.jar org.antlr.Tool DComments.g
javac -cp antlr-3.2.jar *.java
java -cp .:antlr-3.2.jar 主要
(对于 Windows,最后一个命令是java -cp .;antlr-3.2.jar Main
:)
产生以下输出:
评论 :: /+ 评论 /+ 和 +/ 评论 +/
评论 :: /+ 评论 +/