1

我想模拟元空间OOM。我打算ClassA通过不同的 URLClassLoader 加载类,这里是代码:

package classloader;

import java.io.File;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.ArrayList;
import java.util.List;

class ClassA {
    public void method(String input){}
}

public class ClassMetadataLeakSimulator {
    private final static int NB_ITERATIONS_DEFAULT = 50000;

    public static void main(String[] args) {
        System.out.println("Class metadata leak simulator");
        int nbIterations = (args != null && args.length == 1) ? Integer.parseInt(args[0]) : NB_ITERATIONS_DEFAULT;
        try {
            List<ClassLoader> list = new ArrayList<>();
            URL url = new File(".").toURI().toURL();
            URL[] urls = new URL[]{url};
            System.out.println(url);
            for (int i = 0; i < nbIterations; i++) {
                URLClassLoader newClassLoader = new URLClassLoader(urls);
                list.add(newClassLoader);
                newClassLoader.loadClass("classloader.ClassA");
            }
        }
        catch (Throwable any) {
            System.out.println("ERROR: " + any);
        }
        System.out.println("Done!");
    }
} 

但是,奇怪的是,加载的类数在达到 jvisualvm 中显示的 1437 时停止增加,并且使用的元空间大小很低,即使 for 循环已经运行了数百万次。似乎每个新的 URLClassLoader 实例都没有加载 ClassA。为什么?

4

1 回答 1

1

问题是URLClassLoader系统类加载器的父级。在加载时,它首先会尝试从父类加载器获取类。所以类只会被加载一次,你不会得到元空间 OOM。

您需要一个独立的类加载器来完成这项工作。另一种选择是使用cglibasm库动态生成类。

loadClass接受不是规范的而是二进制 的类名。

于 2016-08-22T14:29:38.567 回答