ANTLR przed v4 miał wewnętrzną obsługę StringTemplate (można określić, że gramatyka jest wyjście ST). Począwszy od wersji 4, to wsparcie wydaje się być porzucone.
Jedną z opcji jest użycie interfejsów odbiornika lub użytkownika do ręcznego tworzenia szablonów. Gość jest prawdopodobnie bardziej przydatny w tym kontekście.
Inną opcją, którą obecnie badam, jest przypisanie parametru ParseTree (wynik z analizy) jako parametru do szablonu. Używam niestandardowej ModelAdapter dla ParserRuleContext więc mogę przejść podrzędne konteksty z szablonów.
Przykład: Zakładam, że używasz gramatyki dla PL/SQL. Wtedy można mieć grupę szablonu takiego:
plsql_block(block) ::= <<{
<declarations(block.declare_section)>
<body(block.body)>
}>>
declarations(ds) ::= "<ds.item_declaration:itemDecl()>"
itemDecl(id) ::= "<id.variable_declaration:varDecl()>"
varDecl(vd) ::= "<vd.datatype.text> <vd.ID>;"
body(b) ::= "<b.text>"
będzie też trzeba ModelAdapter dla ParserRuleContext (to tylko przykład jedynej metody w nim):
@Override
public Object getProperty(Interpreter interpreter, ST seld, Object o, Object property, String propertyName) throws STNoSuchPropertyException
{
Method m = null;
try {
String mn = "get" + Character.toUpperCase(propertyName.charAt(0)) + propertyName.substring(1);
m = o.getClass().getMethod(mn);
} catch (Exception e) {
}
if (m == null)
try {
m = o.getClass().getDeclaredMethod(propertyName);
} catch (Exception e) {
}
if (m != null) try {
return m.invoke(o);
} catch (Exception e) {
throw new STNoSuchPropertyException(e, property, propertyName);
}
else
throw new STNoSuchPropertyException(null, property, propertyName);
}
Teraz może wykonać następujące czynności:
ANTLRInputStream input = new ANTLRInputStream(new FileInputStream("block_test.sql"));
PLSQLLexer lexer = new PLSQLLexer(input);
CommonTokenStream tokens = new CommonTokenStream(lexer);
PLSQLParser parser = new PLSQLParser(tokens);
parser.setBuildParseTree(true);
ParseTree tree = parser.plsql_block();
STGroupFile stg = new STGroupFile("test.stg");
stg.registerModelAdaptor(ParserRuleContext.class, new ContextModelAdapter());
ST t = stg.getInstanceOf("plsql_block");
t.add("block", tree);
System.out.println(t.render());
Mam nadzieję, że to pomoże!
Możesz podać przykład, na przykład fragment gramatyki/szablonu, który działa z antlr2/3, ale nie z antlr4? Czy używasz parsego porcelli, czy też sam piszesz? –
Główna różnica między gramatyką ANTLR3 i ANTLR 4 'polega na usunięciu predykatów składniowych. Patrz: http://stackoverflow.com/questions/18431158/syntactic-predicates-upgrading-from-antlr-3-to-antlr-4. Nie, napisałem własny parser, Ref: https://github.com/developeron29/Antlr-4-PLSQL-Parser---AST-Generator/ –
czy użyłeś istniejącego pliku gramatycznego kompatybilnego z v4? Jeśli tak, możesz udostępnić link –