![](/img/trans.png)
[英]Hung Threads java.lang.ClassLoader.findBootstrapClass
[英]Java ClassLoader keeps creating more threads
我的程序的整体概念涉及从.class文件加载插件,运行它,关闭它,手动更新.class文件,最后重新打开它。 目前它
URLClassLoader
加载.class文件并开始执行main方法。 ScheduledThreadPoolExecutor
),它定期查询服务。 ScheduledThreadPoolExecutor
上调用.shutdown()
,所有线程都死掉了。 URLClassLoader
上调用.close()
并将URLClassLoader
变量设置为null。 这些步骤中的所有内容都有效,新的.class文件按预期工作。
当插件再次启动时,我遇到的问题就开始了。 每次运行重启过程时,它都会产生一个额外的插件实例。
第一次运行:1个运行插件
第二轮:2个正在运行的插件
第3次运行:3个运行插件
等等
我觉得很奇怪的是,它并没有在每次启动时产生两倍的插件,它只会产生一个额外的插件。 额外的代码只能再执行一次,而不是每个先前的线程。 每次后续第二次调用startup()
,它会创建一个新的URLClassLoader
(旧的URLClassLoader
是关闭的还是否定的),也会以某种方式启动所有过去的URLClassLoaders
? 尝试在调试模式下运行它,并且它不跟踪每个活动线程。 我需要维护以前的URLClassLoader
,而不删除对后台运行的现有对象的任何引用。
鉴于该计划的复杂性,很难给予SSCCE。
public class PluginHandler
{
private static URLClassLoader cl = null;
private static String = "somedir";
public void restart()
{
(new Thread() {
public void run() {
if (cl != null) {
cl.invokeClass(pckg, "main", "-shutdown");
cl.close();
cl = null;
}
try {
Thread.sleep(15000);
} (catch InterruptedException e) {
System.out.println("Interrupted");
}
cl = URLClassLoader cl = new URLClassLoader(new URL[] { new File(path).toURI().toURL() } ));
cl.invokeClass(pckg, "main", "-startup");
}).start();
}
public URLClassLoader invokeClass(String pckgName, String mthdName, String[] args)
throws Exception
{
Class<?> loadedClass = cl.loadClass(pckgName);
Method m = loadedClass.getMethod(mthdName, args.getClass());
m.invoke(null, new Object[] { args });
return urlcl;
}
}
public class PluginMain
{
public static void main(String[] args) {
if (args[0].equals("-startup") {
new PluginController.run();
}
else if (args[0].equals("-shutdown") {
PluginController.shutdown();
}
}
}
public class PluginController implements Runnable
{
static ScheduledThreadPoolExecutor st;
static ScheduledFuture<?> sf;
public void run() {
st = new Scheduled ThreadPoolExecutor(1);
sf = st.scheduleWithFixedDelay(new Plugin(), 0, 10, Time_Unit.SECONDS);
sf.wait();
System.out.println("Returns from run()"); //prints after shutdown is run.
}
public static void shutdown() {
sf.cancel();
st.shutdown();
}
}
public class Plugin implements Runnable
{
public void run() {
//Queries some service
}
}
编辑:所有插件同时运行相同的代码行。 我提到了那些睡眠,我怀疑它会使不同的线程完全同步。
Mark W的建议引导我使用jmap和其他过程分析程序找出究竟发生了什么。 那里有很多有用的工具。 通过VisualVM和Eclipse Memory Analyzer(MAT)的结合, 我能够发现我没有为我的日志关闭FileHandler。 这么多小时下来了!
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.