ANTLR4解析器中的访问者文件问题



Visitor类中要写什么?

我们已经为我们的语言制定了语法。我们不需要对它执行任何操作。如果语言是通过书面语法传递的,那么我们只想从中获取一些对象。

给定语言作为输入:

Dec 17 14:00:00 103.56.229.11 firewall,info FFFW forward: in:<pppoe-mm.demo.649> out:sfp-sfpplus1.vlan113, proto TCP (ACK,PSH), 10.0.15.245:49831->103.235.46.39:443, NAT (10.0.15.245:49831->202.173.127.253:49831)->103.235.46.39:443, len 250

期望输出:

Dec, 17, 14:00:00, 103.56.229.11, pppoe-mm.demo.649, TCP, 10.0.15.245:49831, 103.235.46.39:443, 202.173.127.253:49831

我们的语法(文件名:sys.g):(运行良好,我们使用ANTLRWorks2进行了验证)

/*
 * To change this license header, choose License Headers in Project Properties.
 * To change this template file, choose Tools | Templates
 * and open the template in the editor.
 */
grammar sys;
r: IDENT NUM time ip x+ user xout proto xuser ipfull xtra ipfull xtra1 ipfull xtra ipfull xtra2 ipfull xtra3; 
time: NUM SEP NUM SEP NUM;
ip: NUM USER NUM USER NUM USER NUM ;
ipfull: NUM USER NUM USER NUM USER NUM SEP NUM ;
x: (IDENT | SEP | NUM)+ LTHAN;
user: (IDENT | USER | NUM)+ ;
xuser: (IDENT | SEP | NUM)+ ;
xout: GTHAN IDENT+ SEP IDENT+ USER IDENT+ USER IDENT SEP IDENT;
proto: IDENT ;
xtra: USER GTHAN ;
xtra1: SEP IDENT SEP;
xtra2: SEP xtra;
xtra3: SEP IDENT NUM;
IDENT: ('a'..'z' | 'A'..'Z')('a'..'z' | 'A'..'Z' | '0'..'9')* ;
NUM: ('0'..'9')+ ;
LTHAN: '<' ;
GTHAN: '>' ;
SEP: ':' | ',' | '(' | ')' ;
USER: '-' | '.' ;
WS : (' ' | 't' | 'r' | 'n')+ -> skip ;

为给定语言生成的树:

为语言生成的树

问题1:我们使用antlr4.5编译了语法文件,还使用了visitor。所以我们的问题是如何在另一个文件中打印特定对象?

问题2:是否需要创建另一个名为"value"的类,将值返回给访问者?

EvalVisitor.java文件:

public class EvalVisitor extends sysBaseVisitor{
//
}

我们的主要java文件,即SysLogCheck.java,其中我们使用语法sys.g文件生成的Lexer(SysLexer.java)和Parser(SysParser.java)。

import org.antlr.v4.runtime.ANTLRFileStream;
import org.antlr.v4.runtime.CommonTokenStream;
import org.antlr.v4.runtime.tree.ParseTree;
import java.io.*;
import org.antlr.v4.runtime.*;
public class SysLogCheck {
    public static void main(String[] args) throws Exception {
        ANTLRInputStream input = new ANTLRInputStream(new FileInputStream(new File("input.txt")));
        sysLexer lexer = new sysLexer(input);
        CommonTokenStream tokens = new CommonTokenStream(lexer);
        sysParser parser = new sysParser(tokens);
        ParseTree tree = parser.r();
        EvalVisitor visitor = new EvalVisitor();
        visitor.visit(tree);        
    }
}

关于您的第一个问题:

下面是一个粗糙访问者的例子,它输出Dec, 17, 14:00:00:

在读取/* do something with the results */的行中,可以放置一些保存结果的代码。

import org.antlr.v4.runtime.tree.ParseTree;
public class EvalVisitor extends sysBaseVisitor{
  class LogEntry {
    String ident1;
    String dayNum;
    String ip;
    /*
    ...
     */
  }
  static LogEntry logEntry;
  @Override
  public Object visit(ParseTree tree) {
    /* Setup logentry used by all visitors (this case, there is only a single visitor...)*/
    logEntry = new LogEntry();
    /* visit */
    final Object o = super.visit(tree);
    /* do something with the results */
    System.out.println(logEntry.ident1 + ", " + logEntry.dayNum + ", " + logEntry.ip);
    return o;
  }
  StringBuilder stringBuilder;
  @Override
  public Object visitR(sysParser.RContext ctx) {
    logEntry.ident1 = ctx.IDENT().getText();
    logEntry.dayNum = ctx.NUM().getText();
    return super.visitR(ctx);
  }
  @Override
  public Object visitTime(sysParser.TimeContext ctx) {
    logEntry.ip = ctx.getText();
    return super.visitTime(ctx);
  }
  @Override
  public Object visitIp(sysParser.IpContext ctx) {
    return super.visitIp(ctx);
  }
  @Override
  public Object visitIpfull(sysParser.IpfullContext ctx) {
    return super.visitIpfull(ctx);
  }
  @Override
  public Object visitX(sysParser.XContext ctx) {
    return super.visitX(ctx);
  }
  @Override
  public Object visitUser(sysParser.UserContext ctx) {
    return super.visitUser(ctx);
  }
  @Override
  public Object visitXuser(sysParser.XuserContext ctx) {
    return super.visitXuser(ctx);
  }
  @Override
  public Object visitXout(sysParser.XoutContext ctx) {
    return super.visitXout(ctx);
  }
  @Override
  public Object visitProto(sysParser.ProtoContext ctx) {
    return super.visitProto(ctx);
  }
  @Override
  public Object visitXtra(sysParser.XtraContext ctx) {
    return super.visitXtra(ctx);
  }
  @Override
  public Object visitXtra1(sysParser.Xtra1Context ctx) {
    return super.visitXtra1(ctx);
  }
  @Override
  public Object visitXtra2(sysParser.Xtra2Context ctx) {
    return super.visitXtra2(ctx);
  }
  @Override
  public Object visitXtra3(sysParser.Xtra3Context ctx) {
    return super.visitXtra3(ctx);
  }
  //
}

最新更新