如何在Eclipse RCP应用程序中使用java.lang.instrument?

5
为了使用JDK 5中引入的仪器化功能,您可以使用传递给JVM的-javaagent标志。这将在静态premain方法中注入一个Instrumentation类的实例。例如,在像这样的类中:
public class MyClass {
    public static Instrumentation inst;
    public static void premain(String options, Instrumentation inst) {
        MyClass.inst = inst;
    }
}

通过适当的清单文件,您可以按如下方式运行它:

 java -javaagent:myfiles.jar SomeClass

这将调用SomeClasspremain方法,然后调用main方法。在Java.SizeOf项目中使用这种方法来猜测一个Java对象的大致大小。
现在,在Eclipse RCP中,每个bundle都有自己的类加载器。这意味着我们存储在MyClass中的静态Instrumentation对于Eclipse应用程序是不可见的。javaagent使用一个类加载器,Eclipse bundles使用另一个类加载器。当我们从插件内部访问MyClass.inst时,它是null,因为那个类不是javaagent加载并调用premain的那个类。
关于可能解决方案的其他线索,请参考rcp邮件列表上的此线程。但没有什么确定的结论。
有没有办法绕过这个问题?在eclipsezone文章中提到的Eclipse-BuddyPolicy听起来不错。我尝试了:
Eclipse-BuddyPolicy: app

在我的插件中没有成功。我需要类似 Eclipse-BuddyPolicy: javaagent 的东西。有什么建议吗?

你有收到任何异常吗?一个陷阱是代理类没有包。(即 MyClass 可能不在默认包中)。它在 Eclipse 外面工作吗? - stacker
没错,在Eclipse之外使用POJO的东西很好用。实际上,premain中的“println”可以在Eclipse中显示,但是当从任何插件调用静态Instrumentation时,它就为null。 - richq
1个回答

5
我认为最简单的解决方案是使用全局属性对象。在pre-main中将Instrumentation对象存储为全局属性,然后从任何地方访问它(属性对象在所有类加载器中都是相同的):

[编辑:已更新]

public class MyClass {
    private static final String KEY = "my.instrumentation";
    public static void premain(String options, Instrumentation inst) {
        Properties props = System.getProperties();
        if(props.get(KEY) == null)
           props.put(KEY, inst);
    }

    public static Instrumentation getInstrumentation() { 
       return System.getProperties().get(KEY);
    }
}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接