我是一名ANTLR初学者,想要计算符号的SHA1-Hash。
我的简化示例语法:
grammar Example;
method @after{calculateSha1($text); }: 'call' ID;
ID: 'A'..'Z'+;
WS: (' '|'\n'|'\r')+ {skip(); }
COMMENT: '/*' (options {greedy=false;}: .)* '*/' {$channel=HIDDEN}
当词法分析器删除所有空格时,不同的字符串callABC
,call /* DEF */ ABC
不幸地得到相同的SHA1-Hash值。
是否可以在开始和结束标记之间获取规则的“原始”文本以及所有跳过的空格和其他通道的文本?
(我想到的一种可能性是成员WS
- 和COMMENT
- lexer规则中的所有字符,但是还有更多规则,所以这不太实用。)
我使用标准的ANTLRInputStream来提供Lexer,但我不知道如何接收原始文本。
答案 0 :(得分:4)
而不是skip()
- ping WS
令牌,也将其放在HIDDEN
频道上:
grammar Example;
@parser::members {
void calculateSha1(String text) {
try {
java.security.MessageDigest md = java.security.MessageDigest.getInstance("SHA-1");
byte[] sha1 = md.digest(text.getBytes());
System.out.println(text + "\n" + java.util.Arrays.toString(sha1) + "\n");
} catch(Exception e) {
e.printStackTrace();
}
}
}
parse
: method+ EOF
;
method
@after{calculateSha1($text);}
: 'call' ID
;
ID : 'A'..'Z'+;
WS : (' ' | '\t' | '\n' | '\r')+ {$channel=HIDDEN;};
COMMENT : '/*' .* '*/' {$channel=HIDDEN;};
上面的语法可以用以下方法测试:
import org.antlr.runtime.*;
public class Main {
public static void main(String[] args) throws Exception {
String source = "call ABC call /* DEF */ ABC";
ExampleLexer lexer = new ExampleLexer(new ANTLRStringStream(source));
ExampleParser parser = new ExampleParser(new CommonTokenStream(lexer));
parser.parse();
}
}
将打印以下内容到控制台:
call ABC [48, -45, 113, 5, -52, -128, -78, 75, -52, -97, -35, 25, -55, 59, -85, 96, -58, 58, -96, 10] call /* DEF */ ABC [-57, -2, -115, -104, 77, -37, 4, 93, 116, -123, -47, -4, 33, 42, -68, -95, -43, 91, 94, 77]
即:相同的解析器规则,但不同的$text
(以及不同的SHA1)。