如何获得最终包名?

时间:2015-04-20 09:38:11

标签: java parsing javaparser

我有代码

private static class MyVisitor extends VoidVisitorAdapter<Object> {
    @Override
    public void visit(MethodCallExpr exp, Object arg) {
        System.out.println("Scope: "  + exp.getScope());
        System.out.println("Method: " + exp.getName());
        if(exp.getArgs() != null)
            for(Expression e : exp.getArgs()) {
                System.out.println("\tArgument: " + e.toString());

            }
        System.out.println();
    }
}

并且

CompilationUnit cu = JavaParser.parse(new File("Test.java"));

for(TypeDeclaration type : cu.getTypes()) {
    for(BodyDeclaration dec : type.getMembers()) {
        if(dec instanceof MethodDeclaration) {
            MethodDeclaration mdec = (MethodDeclaration) dec;
            BlockStmt block = mdec.getBody();
            for(Statement stmt : block.getStmts()) {
                MyVisitor visitor = new MyVisitor();
                s.accept(visitor, null);
            }
        }
    }
}

我有两个问题:

  1. 如何将Statement转换为Expression?我想查一下方法调用。我试图通过这种方式检查它们是不兼容的类型:if(stmt instanceof MethodCallExp)但是出错了。我输入了我现在使用的代码。这是执行检查的一种方式吗?
  2. System.out.println("Scope: " + exp.getScope());将显示对象/包/包+类名。但是如何获得它的类型? 例如代码System.out.println("Hello world"); 应该输出
  3.   

    输入:java.io.PrintStream

2 个答案:

答案 0 :(得分:0)

  1. 语句与表达式之间没有1:1的关系。例如。 ExpressionStmt只包含一个Expression,但WhileStmt包含一个条件Expression和一个包含另一个Statement的主体。
  2. getScope()调用返回的FieldAccessExpr中没有包含类型信息。 JavaParser似乎没有提供获取类型的简单方法。使用Node层次结构及其类型,您可以使用相应的Reflection类来实现目标。对于你的例子,这意味着找到最顶层的范围(类型将是NameExpr)并获得类对象(认识到System是java.lang.System的简称),对于下一个节点(类型将是FieldAccessExpr),你将获得Field该类,然后可以得到类型:
    Class.forName("java.lang.System").getField("out").getType();
    在上面的代码示例中,您将访问者传递给方法体内的每个语句。 如果您在循环之外创建访问者,它将能够记住在先前语句中看到的信息。 这样,您可以向访问者添加一个字段来存储变量声明:
    final HashMap varNameToType = new HashMap();
    您可以存储如下所示的变量声明。 当您进行方法调用时,您可以使用范围尝试通过地图获取类型。如果映射不包含Type,则可能需要使用Reflection,因为该变量可能是JDK类的一部分。我并不是说这涵盖了所有情况。例如。当您访问父类的字段时,您将无法在代码中的任何位置获得有关该字段的任何信息 - 您可能需要查看不同的编译单元。
  3. @Override
    public void visit(
            final VariableDeclarationExpr n,
            final Object arg) {
        final Type varType = n.getType();
        n.getVars().forEach(vd ->
            varNameToType.put(vd.getId().getName(),varType)
        );
    }
    

答案 1 :(得分:0)

如果您不需要坚持JavaParser,也许您可​​以通过Android lint工具的方式来解决它。 EcjParserTest.java。它使用引擎盖下的Lombok

在第380行,它会在测试中将输出与您想要的输出[ExpressionStatement], type: void, resolved method: println java.io.PrintStream进行比较。

我没有更深入的了解,但它看起来是一个很有希望的起点