【问题标题】:How can I use java.lang.instrument in an Eclipse RCP application?如何在 Eclipse RCP 应用程序中使用 java.lang.instrument?
【发布时间】:2011-01-05 01:12:16
【问题描述】:

为了使用 JDK 5 中引入的检测功能,您可以使用传递给 JVM 的 -javaagent 标志。这会将 Instrumentation 类的实例注入到静态 premain 方法中。例如在这样的类中:

public class MyClass {
    public static Instrumentation inst;
    public static void premain(String options, Instrumentation inst) {
        MyClass.inst = inst;
    }
}

使用适当的清单文件,您可以按如下方式运行:

 java -javaagent:myfiles.jar SomeClass

这会调用 premain 方法,然后从 SomeClass 调用 main。在Java.SizeOf Project 中使用了这种方法来猜测 Java 对象的大致大小。

好的,现在在 Eclipse RCP each bundle has its own classloader 中。这意味着我们存储在 MyClass 中的静态 Instrumentation 对 Eclipse 应用程序不可见。 javaagent 使用一个类加载器,Eclipse 包使用另一个加载。当我们从插件中访问MyClass.inst 时,它是null,因为那个 类与javaagent 加载并调用premain 的类不同。

关于可能的解决方案的其他线索是 rcp 邮件列表上的this thread。但尚无定论。

有没有办法解决这个问题? eclipsezone 文章中暗示的Eclipse-BuddyPolicy 听起来不错。我试过了:

Eclipse-BuddyPolicy: app

在我的插件中没有运气。我需要像Eclipse-BuddyPolicy: javaagent 这样的东西。有什么想法吗?

【问题讨论】:

  • 您收到任何例外吗?一个陷阱是没有包的代理类。 (即 MyClass 可能不在默认包中)。它在日食之外工作吗?
  • 是的,在 Eclipse 之外使用 POJO 东西可以正常工作。事实上,premain 中的“println”是从 Eclipse 中显示出来的,但是当从任何插件调用时,静态 Instrumentation 为空。

标签: java eclipse-rcp instrumentation


【解决方案1】:

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

[编辑:更新]

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);
    }
}

【讨论】:

    最近更新 更多