我有一项任务要模拟OutOfMemory: Metaspace错误。我尝试了不同的方法,但没有给我必要的结果。有人能为此目的树立好的榜样吗?
我的第一次尝试(使用javassist):
static ClassPool classPool = ClassPool.getDefault();
@SneakyThrows
public void task() {
try {
for (int i = 0; ; i++) {
Class cl = classPool.makeClass("task1.Test" + i).toClass();
}
} catch (Error er) {
Runtime.getRuntime().gc();
log.error(er.getMessage());
}
}和gradle.properties文件中的设置:
org.gradle.jvmargs=-XX:MaxMetaspaceSize=70M
但我有个错误:
异常:在“java.lang.OutOfMemoryError”线程中从UncaughtExceptionHandler抛出的java.lang.OutOfMemoryError
但我抓不到。
我的第二次尝试:
public void task() {
try {
URL url = new File("D:/classes").toURI().toURL();
URL[] urls = {url};
ClassLoadingMXBean loadingMXBean = ManagementFactory.getClassLoadingMXBean();
List<ClassLoader> classLoaders = new ArrayList<>();
while (true) {
ClassLoader classLoader = new URLClassLoader(urls);
classLoaders.add(classLoader);
classLoader.loadClass("test1.Test1");
log.info("Total: " + loadingMXBean.getTotalLoadedClassCount());
log.info("Active: " + loadingMXBean.getLoadedClassCount());
log.info("Unloaded: " + loadingMXBean.getUnloadedClassCount());
}
} catch (Exception ex) {
log.error(ex + ex.getMessage());
}
}但这不管用,这个想法告诉我有法律记忆。
发布于 2021-07-15 20:50:39
您忘记了JVM中的类加载是如何工作的。类加载器将首先尝试让它的父类加载您正在寻找的类。为了使测试工作,您需要一个自定义类加载器,它不会为您正在测试的一个类委托它的父类。
下面是一个在Java 8中工作的示例
DummyClass.java (默认包)
public class DummyClass {
static String padding = "1234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890";
}Test.java (默认包)
import java.lang.management.ClassLoadingMXBean;
import java.lang.management.ManagementFactory;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.ArrayList;
import java.util.List;
public class Test {
public static void main(String[] args) {
try {
URL url = Test.class.getResource("").toURI().toURL();
System.out.println("URL = " + url);
URL[] urls = {url};
ClassLoadingMXBean loadingMXBean = ManagementFactory.getClassLoadingMXBean();
List<ClassLoader> classLoaders = new ArrayList<>();
System.out.println("Total: " + loadingMXBean.getTotalLoadedClassCount());
System.out.println("Active: " + loadingMXBean.getLoadedClassCount());
System.out.println("Unloaded: " + loadingMXBean.getUnloadedClassCount());
int i = 0;
while ( true ) {
i++;
System.out.println("### Iteration " + i + " ###");
ClassLoader classLoader = new URLClassLoader(urls) {
@Override
public Class<?> loadClass(String name) throws ClassNotFoundException {
final Class<?> loadedClass;
if ( "DummyClass".equals(name) ) {
System.out.println (this + " - finding class " + name);
loadedClass = findClass(name);
System.out.println (this + " - loading class " + loadedClass);
} else {
// delegate to parent
loadedClass = this.getParent().loadClass(name);
}
return loadedClass;
}
};
classLoaders.add(classLoader);
classLoader.loadClass("DummyClass");
System.out.println("Total: " + loadingMXBean.getTotalLoadedClassCount());
System.out.println("Active: " + loadingMXBean.getLoadedClassCount());
System.out.println("Unloaded: " + loadingMXBean.getUnloadedClassCount());
}
} catch ( Exception ex ) {
ex.printStackTrace();
}
}
}与-XX:MaxMetaspaceSize=100m一起运行时,我最终得到了以下内容:
> ### Iteration 16318 ###
> Test$1@531ed68e - finding class DummyClass
> Test$1@531ed68e - loading class class DummyClass
> Total: 16747 Active:
> 16747 Unloaded: 0
> ### Iteration 16319 ###
> Test$1@6bbd4048 - finding class DummyClass
> Exception in thread "main" java.lang.OutOfMemoryError: Metaspace at
> java.lang.ClassLoader.defineClass1(Native Method) at
> java.lang.ClassLoader.defineClass(ClassLoader.java:756) at
> java.security.SecureClassLoader.defineClass(SecureClassLoader.java:142)
> at java.net.URLClassLoader.defineClass(URLClassLoader.java:468) at
> java.net.URLClassLoader.access$100(URLClassLoader.java:74) at
> java.net.URLClassLoader$1.run(URLClassLoader.java:369) at
> java.net.URLClassLoader$1.run(URLClassLoader.java:363) at
> java.security.AccessController.doPrivileged(Native Method) at
> java.net.URLClassLoader.findClass(URLClassLoader.java:362) at
> Test$1.loadClass(Test.java:42) at Test.main(Test.java:53)https://stackoverflow.com/questions/68399943
复制相似问题