Java:方法中使用的列表字段

时间:2016-03-25 09:12:51

标签: java reflection java-bytecode-asm bytecode-manipulation

在Java中,如何获取方法中使用的字段?

基本上,这与this one in .NET的问题相同。我不想列出类中的字段,而是列出在类的给定方法中使用的字段。

示例:

public class A {
 int a;
 int b;

public int bob(){
 return a-b;
}

我想得到这样的字段:

Fields[] fields = FieldReader.(A.class.getMethod("bob"));

fields[0]=A.afields[1]=A.b

我没有找到任何使用标准反射的解决方案。你认为像ASM这样的字节码操作库是一种方法吗?

2 个答案:

答案 0 :(得分:2)

以下是javassist的示例(您需要将其添加为依赖项,具体取决于您的依赖项管理器首选项)。

此代码列出了public void doSomething();方法中正在访问的字段。

package bcm;

import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.PrintStream;

import javassist.CannotCompileException;
import javassist.ClassPool;
import javassist.CtClass;
import javassist.CtMethod;
import javassist.NotFoundException;
import javassist.bytecode.InstructionPrinter;

public class Person {

    String name;
    String surname;
    int age;

    boolean candrink = false;

    public Person(String name, String surname, int age) {
        super();
        this.name = name;
        this.surname = surname;
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getSurname() {
        return surname;
    }

    public void setSurname(String surname) {
        this.surname = surname;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    public void doSomething() {
        if (this.age > 18) {
            candrink = true;
        }
    }

    public static void main(String[] args) throws IOException,
            CannotCompileException {
        ClassPool pool = ClassPool.getDefault();
        try {
            CtClass cc = pool.get("bcm.Person");
            CtMethod m = cc.getDeclaredMethod("doSomething", null);
            ByteArrayOutputStream baos = new ByteArrayOutputStream();
            PrintStream ps = new PrintStream(baos);
            InstructionPrinter i = new InstructionPrinter(ps);
            i.print(m);
            String content = baos.toString();

            for (String line : content.split("\\r?\\n")) {
                if (line.contains("getfield")) {
                    System.out.println(line.replaceAll("getfield ", ""));
                }
            }

        } catch (NotFoundException e) {
            e.printStackTrace();
        }
    }

}

HTH

答案 1 :(得分:1)

加载类节点后,这变得异常简单。

// Assuming you have loaded classNode
for (MethodNode method : classNode.methods){
    for (AbstractInsnNode ain : method.instructions.toArray()) {
        if (ain.getType() == AbstractInsnNode.FIELD_INSN) {
            FieldInsnNode fin = (FieldInsnNode) ain;
            //fin.name = Field name
            //fin.owner = ClassNode's name
        }
    }
}

Plus ASM比Javassist等库快得多。