如何找出谁在Java中创建线程?

在tomcat中,如果webapp确实停止了none守护进程线程,则无法通过shutdown.sh关闭tomcat

例如:

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关闭。从jvisualvm,线程检查器说:

"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类创建了线程,我的问题是如何从许多Web应用程序中找出谁创建了线程。

谢谢!

回答:

以下是方法的列表,从最快/最可靠到最慢/最困难:

  1. 如果您有该类的源代码,请在构造函数中创建一个异常(实际上不抛出异常)。当您需要了解线程创建的时间时,可以简单地检查或打印它。

  2. 如果没有源,则可以很好地提示线程名称是创建线程的人。

  3. 如果名称暗示了通用服务(如java.util.Timer),则可以在构造函数中的IDE中创建条件断点。条件应为线程名称;当有人使用此名称创建线程时,调试器将停止。

  4. 如果您没有太多线程,请在的构造函数中设置一个断点Thread

  5. 如果线程很多,请将调试器附加到应用程序并将其冻结。然后检查堆栈跟踪。

  6. 如果其他所有方法都失败了,请获取Java运行时的源代码,并在要观察的类中添加日志记录代码,编译一个新的rt.jar并将原始的替换为您的版本。请不要在生产中尝试此方法。

  7. 如果没有问题,可以使用诸如Compuware APM之类的动态跟踪工具,或者,如果您使用的是Linux或Solaris,则可以分别尝试使用SystemTap和dtrace。

以上是 如何找出谁在Java中创建线程? 的全部内容, 来源链接: utcz.com/qa/411285.html

回到顶部