7

在 tomcat 中,如果一个 webapp 确实停止了一个无守护线程,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关闭。来自 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 类创建了线程,我的问题是如何从许多 webapps 中找出谁创建了线程。

谢谢!

4

4 回答 4

22

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

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

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

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

  4. 如果您没有太多线程,请在Thread.

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

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

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

于 2013-09-25T08:29:21.037 回答
2

如果您可以控制该类,则可以在创建它时捕获堆栈跟踪:

public class Test extends TimerTask {
  final StackTraceElement[] callerStack;

  public Test () {
    callerStack = Thread.currentThread().getStackTrace();
  }

  @Override
  public void run() {
    System.out.println("AAAA");
    System.out.println("Creator: "+Arrays.asList(callerStack));
  }

  @Override
  public String toString () {
    return "Test created by "+Arrays.asList(callerStack);
  }
}
于 2013-09-25T08:10:55.007 回答
2

您还可以编写一个小型 java Instrumentation 代理来包装线程构造函数(并在那里进行堆栈转储),而不是尝试修改 rt.jar 类。

于 2016-01-31T21:23:51.247 回答
0

如果您有 Eclipse 或 Intellij 之类的 IDE,则可以在各种 Thread 构造函数中设置断点。您可以使用调试器远程附加到进程。构建线程时,调试器将停止进程。您可以观察诸如线程名称之类的内容。如果要匹配特定线程名称,还可以使用带条件的断点。

设置断点。 https://www.ibm.com/developerworks/library/os-ecbug/

远程调试 https://dzone.com/articles/a-practical-guide-to-java-remote-debugging-in-the

于 2018-11-27T05:01:30.763 回答