我有以下代码生成(故意)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,混合模式)
答案 0 :(得分:1)
所有类加载器都会对它们加载的所有类保持强引用。在您的示例中,您不断重用DynamicClassLoader
的单个实例。这个类加载器反过来保持对您加载的所有类的强引用。因此,垃圾收集器永远不会看到它可以收集的未引用对象。
如果您修改测试用例以使用单独的类加载器,则垃圾收集器应该能够识别您的类未被使用并且应该回收内存。