PermGen的标志无法按预期工作:-XX:+ CMSClassUnloadingEnabled和-XX:+ CMSPermGenSweepingEnabled

时间:2016-12-16 12:05:58

标签: java out-of-memory java-7 openjdk permgen

我有以下代码生成(故意)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,至少可以看到部分改进。但事实并非如此。

问题:我是否误解了这些旗帜的含义?你能详细说明是否属于这种情况吗?否则,有什么建议吗?

N.B。 java -version的输出为:

  

java版本" 1.7.0_95"

     

OpenJDK运行时环境(IcedTea 2.6.4)(7u95-2.6.4-3)

     

OpenJDK 64位服务器VM(内置24.95-b01,混合模式)

1 个答案:

答案 0 :(得分:1)

所有类加载器都会对它们加载的所有类保持强引用。在您的示例中,您不断重用DynamicClassLoader的单个实例。这个类加载器反过来保持对您加载的所有类的强引用。因此,垃圾收集器永远不会看到它可以收集的未引用对象。

如果您修改测试用例以使用单独的类加载器,则垃圾收集器应该能够识别您的类未被使用并且应该回收内存。