Java:你如何真正使用JVMTI的ForceGargabeCollection强制GC?

2022-09-01 21:21:26

我不是在寻找通常的“你只能使用System.gc()在Java中提示GC”的答案,这根本不是这个问题的内容。

我的问题不是主观的,而是基于现实:GC可以在Java中强制使用。我们每天使用的许多程序都这样做:IntelliJ IDEA,NetBeans,VisualVM。

它们都可能迫使GC发生。

它是如何完成的?

我认为他们都使用JVMTI,更具体地说是ForceGarbageCollection(注意“Force”),但我该如何亲自尝试呢?

http://java.sun.com/javase/6/docs/platform/jvmti/jvmti.html#ForceGarbageCollection

另请注意,这个问题不是关于“为什么”我想这样做:“为什么”可能是“好奇心”或“我们正在编写类似于VisualVM的程序”,等等。

问题真的是“如何使用JVMTI的ForceGarbageCollection强制GC”?

JVM是否需要使用任何特殊参数启动?

是否需要任何 JNI?如果是这样,究竟是什么代码?

它是否仅适用于 Sun VM?

任何完整和可编纂的例子都是非常受欢迎的。


答案 1

NetBeans 至少使用 System.gc(): http://hg.netbeans.org/main/annotate/9779f138a9c9/openide.actions/src/org/openide/actions/GarbageCollectAction.java(这是用于显示当前堆并允许您启动 GC 的小按钮)。如果您点击该链接,您将看到它们显式运行终结器。如果您有几千兆的可用磁盘空间,并且想要自己调查代码,则可以通过Mercurial获得:hg clone http://hg.netbeans.org/main/

据我所知,“System.gc()只是一个提示”教条起源于对JLS和JVM Spec的迂腐解释,它允许没有垃圾回收堆的Java实现。那,以及对JavaDoc的不完整解读:

调用 gc 方法表明 Java 虚拟机花费精力回收未使用的对象,以便使它们当前占用的内存可供快速重用。当控件从方法调用返回时,Java 虚拟机已尽最大努力从所有丢弃的对象中回收空间。

读第二句话:“尽最大努力回收空间”比“暗示”强得多。

也就是说,很少有理由打电话给.向高德纳道歉:System.gc()

我们应该忘记内存管理,比如说大约97%的时间:显式垃圾回收是万恶之源


答案 2

我构建了一个基本的java代理,允许调用jvmti ForceGarbageCollection函数:

#include <stdlib.h>
#include <stdio.h>
#include <jvmti.h>


typedef struct {
 jvmtiEnv *jvmti;
} GlobalAgentData;

static GlobalAgentData *gdata;

JNIEXPORT jint JNICALL Agent_OnLoad(JavaVM *jvm, char *options, void *reserved)
{
  printf("load garbager agent\n");
  jvmtiEnv *jvmti = NULL;

  // put a jvmtiEnv instance at jvmti.
  jint result = jvm->GetEnv((void **) &jvmti, JVMTI_VERSION_1_1);
  if (result != JNI_OK) {
    printf("ERROR: Unable to access JVMTI!\n");
  }

  // store jvmti in a global data
  gdata = (GlobalAgentData*) malloc(sizeof(GlobalAgentData));
  gdata->jvmti = jvmti;
  return JNI_OK;
}


extern "C"
JNIEXPORT void JNICALL Java_Garbager_forceGarbageCollection(JNIEnv *env, jclass thisClass) 
{
  printf("force garbage collection\n");
  gdata->jvmti->ForceGarbageCollection();
}

此代理通过 JNI 调用:

class Garbager {
    public static void main(String[] args) {
        Garbager.garbageMemory();
    }

    static void garbageMemory() {
        forceGarbageCollection();
    }

    private static native void forceGarbageCollection();
}

要在 MacOSX 上编译代理:

clang -shared -undefined dynamic_lookup -o garbager-agent.so -I /Library/Java/JavaVirtualMachines/jdk1.8.0_40.jdk/Contents/Home/include/ -I /Library/Java/JavaVirtualMachines/jdk1.8.0_40.jdk/Contents/Home/include/darwin garbager-agent.cpp

要启动 :Garbager

java -agentpath:garbager-agent.so Garbager

基于本教程:拥有堆:使用 JVMTI 迭代类实例