我有代码
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);
}
}
}
}
我有两个问题:
if(stmt instanceof MethodCallExp)
但是出错了。我输入了我现在使用的代码。这是执行检查的一种方式吗?System.out.println("Scope: " + exp.getScope());
将显示对象/包/包+类名。但是如何获得它的类型?
例如代码System.out.println("Hello world");
应该输出输入:java.io.PrintStream
答案 0 :(得分:0)
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)
);
}
答案 1 :(得分:0)
如果您不需要坚持JavaParser
,也许您可以通过Android lint工具的方式来解决它。 EcjParserTest.java。它使用引擎盖下的Lombok
。
在第380
行,它会在测试中将输出与您想要的输出[ExpressionStatement], type: void, resolved method: println java.io.PrintStream
进行比较。
我没有更深入的了解,但它看起来是一个很有希望的起点