在tomcat中,如果web应用程序确实停止了非守护线程,则tomcat无法通过shutdown.sh关闭
例如:
public class demo implements ServletContextListener{
public void contextDestroyed(ServletContextEvent arg0) {
// TODO Auto-generated method stub
// yes,we can cancel timer in here,but this is not the major problem
}
public void contextInitialized(ServletContextEvent arg0) {
Timer timer = new Timer();
timer.schedule(new Test(), 1000, 1000*10);
}
}
public class Test extends TimerTask{
@Override
public void run() {
System.out.println("AAAA");
}
}
如上,tomcat 无法通过 shutdown.sh 关闭。
来自 VisualVM,线程检查器说:
"Timer-0" - Thread t@40
java.lang.Thread.State: TIMED_WAITING
at java.lang.Object.wait(Native Method)
- waiting on <3957edeb> (a java.util.TaskQueue)
at java.util.TimerThread.mainLoop(Timer.java:552)
at java.util.TimerThread.run(Timer.java:505)
Locked ownable synchronizers:
- None
堆栈信息没有指出哪个java类创建了线程,
我的问题是如何找出谁从许多网络应用程序中创建了该线程。
Thanks!
以下是方法列表,从最快/最可靠到最慢/最难排序:
如果您有该类的源代码,请在构造函数中创建一个异常(而不实际抛出它)。当您需要知道线程何时创建时,您可以简单地检查或打印它。
如果您没有源代码,线程名称可以很好地提示创建它的人。
如果名称暗示通用服务(例如java.util.Timer
),然后您可以在 IDE 的构造函数中创建条件断点。条件应该是线程名称;当有人使用该名称创建线程时,调试器将停止。
如果没有太多线程,请在构造函数中设置断点Thread
.
如果您有很多线程,请将调试器附加到应用程序并冻结它。然后检查堆栈跟踪。
如果其他一切都失败了,请获取Java 运行时的源代码并在您想要观察的类中添加日志记录代码,编译一个新的rt.jar
并将原来的版本替换为您的版本。请不要在生产中尝试这个。
如果钱不是问题,您可以使用动态跟踪工具,例如康威APM或者,如果您使用的是 Linux 或 Solaris,您可以尝试系统点击和 dtrace 分别。
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)