我有以下代码(有意)生成 PermGen java.lang.OutOfMemoryError
:
public class Main {
public static void main(String[] args) throws InstantiationException, IllegalAccessException, InterruptedException {
String name = "MyClass";
DynamicClassLoader cl = new DynamicClassLoader();
int i = 0;
while (true) {
//code for generating the binary for a class to be loaded.
ClassWriter cw = new ClassWriter(ClassWriter.COMPUTE_FRAMES);
cw.visit(Opcodes.V1_7, Opcodes.ACC_PUBLIC, name + ++i, null, "java/lang/Object", null);
MethodVisitor con = cw.visitMethod(Opcodes.ACC_PUBLIC, "", "()V", null, null);
con.visitCode();
con.visitVarInsn(Opcodes.ALOAD, 0);
con.visitMethodInsn(Opcodes.INVOKESPECIAL, "java/lang/Object", "", "()V", false);
con.visitInsn(Opcodes.RETURN);
con.visitMaxs(1, 1);
cw.visitEnd();
//binary code for class successfully generated
Class clazz = cl.defineClass(name + i, cw.toByteArray());
Object o = clazz.newInstance();
System.out.println(o.getClass().getName());
}
}
private static class DynamicClassLoader extends ClassLoader {
public Class defineClass(String name, byte[] b) {
return defineClass(name, b, 0, b.length);
}
}
}
我在 Java 7 中运行此代码。正如预期的那样,它收到 java.lang.OutOfMemoryError: PermGen space
错误。
当我尝试使用上述标志运行该程序时,如下所示:
java -XX:+CMSClassUnloadingEnabled -XX:+CMSPermGenSweepingEnabled -jar target/permgen.jar
,我仍然遇到相同的错误,与我在没有标志的情况下运行它时完全相同。 我预计,如果我设置这些标志,即使不会发生 PermGen 的完全刷新,至少也会看到部分改进。但事实并非如此。
问题:我是否误解了这些标志的含义?如果是这样的话,能详细说一下吗?否则,有什么建议吗?
注意java -version
的输出是:
java version "1.7.0_95"
OpenJDK Runtime Environment (IcedTea 2.6.4) (7u95-2.6.4-3)
OpenJDK 64-Bit Server VM (build 24.95-b01, mixed mode)
最佳答案
所有类加载器都保留对其加载的所有类的强引用。在您的示例中,您继续重用DynamicClassLoader
的单个实例。该类加载器又保留对您加载的所有类的强引用。因此垃圾收集器永远看不到它可以收集的未引用对象。
如果您修改测试用例以使用单独的类加载器,垃圾收集器应该能够识别出您的类没有被使用,并且应该回收内存。
关于out-of-memory - PermGen 的标志未按预期工作 : -XX:+CMSClassUnloadingEnabled and -XX:+CMSPermGenSweepingEnabled,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41184036/