使用ASM获取方法参数

时间:2013-05-24 20:09:03

标签: java reflection java-bytecode-asm

我找到了一些示例,向我展示了使用MethodAdapter的某些方法调用的位置:

public void visitMethodInsn(int opcode, String owner, String name, String desc) {
        if (owner.equals(targetClass)
                && name.equals(targetMethod.getName())
                && desc.equals(targetMethod.getDescriptor())) {
            callsTarget = true;
        }
  }

我需要这些参数,例如,如果我有object.getText("mykey")我想获得文本“mykey”。

这可能吗?

1 个答案:

答案 0 :(得分:0)

我有一个framework您可能会觉得有用(具体来说,procyon-compilertools)。它将使您能够以比ASM更面向对象的方式执行您的要求。但是,该项目仍处于开发阶段,可能会发生变化,因此我不建议在生产项目中使用它。

您可以将此作为起点:

public class CallInspectionSample {
    static class Target {
        public static void main(String[] args) {
            new Target().getText("MyKey");
        }

        public String getText(final String key) {
            return null;
        }
    }

    public static void main(String[] args) {
        final TypeReference targetType = MetadataSystem.instance().lookupType("CallInspectionSample$Target");
        final TypeDefinition resolvedType = targetType.resolve();
        final MethodDefinition mainMethod = resolvedType.getDeclaredMethods().get(1);
        final MethodDefinition getTextMethod = resolvedType.getDeclaredMethods().get(2);

        final MethodBody mainBody = mainMethod.getBody();

        final Block methodAst = new Block();
        final DecompilerContext context = new DecompilerContext();

        context.setCurrentType(resolvedType);
        context.setCurrentMethod(mainMethod);

        methodAst.getBody().addAll(AstBuilder.build(mainBody, true, context));

        AstOptimizer.optimize(context, methodAst);

        for (final Expression e : methodAst.getChildrenAndSelfRecursive(Expression.class)) {
            if (e.getCode() == AstCode.InvokeVirtual &&
                ((MethodReference) e.getOperand()).resolve() == getTextMethod) {

                // Analyze arguments here (skip first for instance methods)...
                System.out.println(e.getArguments());
            }
        }
    }
}

(示例输出[initobject:Target(Target::<init>), ldc:String("MyKey")]