java - 如何检索方法内的任何内容

标签 java methods constructor local-variables bcel

据我所知,java无法检索方法内的任何内容。所以我在 javac 中使用选项 -g 或 -g:vars。

例如:

class Test {
    int a=0;
    void method(boolean boo){
        String b;
        try
        {
            new Thread().sleep(1000);
        }
        catch(InterruptedException e){}
        JOptionPane.showMessageDialog(null,"test");
        BufferedImage image=ImageIO.read(new File("C:\\file.png"));
    }
}

所以,我使用 BCEL 来检索局部变量。

import org.apache.bcel.classfile.*;
import org.apache.bcel.Repository;
class javap
{
    public static void main(String[]args)
    {
        try
        {
            JavaClass jc = Repository.lookupClass("test");
            ConstantPool constantPool = jc.getConstantPool();
            Method [] method=jc.getMethods();
            for (Method m : method) 
            {
                LocalVariableTable lvt=m.getLocalVariableTable();
                LocalVariable[] lv=lvt.getLocalVariableTable();
                for(LocalVariable l : lv)
                {   
                    System.out.println(l.getName()+" : "+l.getSignature());
                }
            }
        }
        catch(Exception ex)
        {
            ex.printStackTrace();
        }
    }
}

但是如果变量没有像String b那样初始化,它就不起作用。此外,我想跟踪诸如 new Thread()new File() 之类的构造函数调用,以及静态方法的调用和 JFileChooser 内部的初始化就像 new FileJOptionPane 一样。所以我想在输出中看到 ThreadString bJOptionPaneImageIOFile.

我应该怎么做,才能将它们打印在我的程序中?

最佳答案

你根本无法获取b变量,因为java编译器(至少是javac和ecj)根本不会将它放入生成的类文件中:如果未分配变量,则不会有变量槽已分配且不存储在 LocalVariableTable 中。您可以创建具有更长名称的未使用变量,例如 String blahblah;,编译类,在文本编辑器中打开编译后的 .class 文件并搜索 blahblah 字符串。你不会找到它。所以BCEL无法帮助你找到缺失的变量。

如果你想跟踪新对象的创建和静态方法的调用,你可以扫描方法字节码。使用 BCEL 执行此操作的最简单方法是利用 MethodGen(即使您不想生成新方法)。完整代码如下:

import org.apache.bcel.Constants;
import org.apache.bcel.Repository;
import org.apache.bcel.classfile.ConstantMethodref;
import org.apache.bcel.classfile.ConstantPool;
import org.apache.bcel.classfile.JavaClass;
import org.apache.bcel.classfile.LocalVariable;
import org.apache.bcel.classfile.LocalVariableTable;
import org.apache.bcel.classfile.Method;
import org.apache.bcel.generic.ConstantPoolGen;
import org.apache.bcel.generic.INVOKESTATIC;
import org.apache.bcel.generic.InstructionHandle;
import org.apache.bcel.generic.MethodGen;
import org.apache.bcel.generic.NEW;

class javap
{
    public static void main(String[]args)
    {
        try
        {
            JavaClass jc = Repository.lookupClass("Test");
            ConstantPool constantPool = jc.getConstantPool();
            Method [] method=jc.getMethods();
            for (Method m : method) 
            {
                LocalVariableTable lvt=m.getLocalVariableTable();
                LocalVariable[] lv=lvt.getLocalVariableTable();
                for(LocalVariable l : lv)
                {   
                    System.out.println(l.getName()+" : "+l.getSignature());
                }
            }
            ConstantPoolGen cpg = new ConstantPoolGen(constantPool);
            for(Method m : method)
            {
                MethodGen mg = new MethodGen(m, m.getName(), cpg);
                for(InstructionHandle ih = mg.getInstructionList().getStart(); 
                        ih != null; ih = ih.getNext())
                {
                    if(ih.getInstruction() instanceof NEW) 
                    {
                        NEW newInst = ((NEW)ih.getInstruction());
                        String className = constantPool.getConstantString(
                            newInst.getIndex(), Constants.CONSTANT_Class);
                        System.out.println("Class instantiation: "+className);
                    }
                    if(ih.getInstruction() instanceof INVOKESTATIC) 
                    {
                        INVOKESTATIC newInst = ((INVOKESTATIC)ih.getInstruction());
                        String className = constantPool.getConstantString(
                                ((ConstantMethodref) constantPool
                                        .getConstant(newInst.getIndex()))
                                        .getClassIndex(),
                                Constants.CONSTANT_Class);
                        System.out.println("Static call: "+className);
                    }
                }
            }
        }
        catch(Exception ex)
        {
            ex.printStackTrace();
        }
    }
}

输出如下:

this : LTest;
this : LTest;
boo : Z
Class instantiation: java/lang/Thread
Static call: java/lang/Thread
Static call: javax/swing/JOptionPane
Class instantiation: java/io/File
Static call: javax/imageio/ImageIO

请注意,您有两次 java/lang/Thread ,因为 new Thread() 被捕获为对象创建和 Thread.sleep() > 被捕获为静态方法调用。

关于java - 如何检索方法内的任何内容,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30994441/

相关文章:

java - 制作无法卸载/删除的Android应用程序

java - 不确定如何创建使用其他类方法的菜单

ubuntu - Ubuntu 方案和 XWin 方案的区别

带有 const 参数的 C++ 构造函数

javascript - 通过原型(prototype)对象或构造函数设置方法,有什么区别?

java - 从单元测试中产生一个单独的进程

java - tomcat配置导致@Service的2个副本运行

java - Java什么时候读取方法字节码?

objective-c - 如何根据 NSArray 的元素选择方法 (Objective-C)

java - 米格布局 : Resize components relatively to a screen size