繁体   English   中英

如何获得最终包名?

[英]How to get final package name?

我有代码

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"); 应该输出

键入:java.io.PrintStream

  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类的一部分。 我并不是说这涵盖了所有情况。 例如,当您访问父类的字段时,您将无法在代码中的位置获得有关该字段的任何信息 - 您可能需要查看不同的编译单元。
@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)
    );
}

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

在第380行,它在测试中比较了一个类似于你想要的输出[ExpressionStatement], type: void, resolved method: println java.io.PrintStream

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

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM