5

[更新:在top我按下显示线程而不是进程的 shift+H 之后,它将 Java 线程显示为 R 并使用 100% CPU 时间,这是我在发布此问题之前所期望的。]

既然一个Java进程有多个线程,每个线程可能处于不同的状态,那么Linuxtop命令是如何确定Java进程状态的呢?

如果我运行以下代码,

public class Test{
  public static void main(String[] args){
     while (true){
       int n = (int)(Math.random() * 1000);
     }
  }
}

然后,运行top显示进程状态为 S,它正在使用 100% 的 CPU 时间。

此外,正在运行的strace节目并且只显示:

futex(0x7f6ba759c9d0, FUTEX_WAIT, 26060, NULL

但是,运行jstack显示主线程是RUNNABLE:

"main" prio=10 tid=0x00007fd7ec007800 nid=0x669b runnable [0x00007fd7f5754000]
  java.lang.Thread.State: RUNNABLE
    at Test.main(Test.java:5)

jstack还表明,只有两个线程处于 WAITING 状态:

"Finalizer" daemon prio=10 tid=0x00007fd7ec080000 nid=0x66a6 in Object.wait() [0x00007fd7f0252000]
    java.lang.Thread.State: WAITING (on object monitor)
        at java.lang.Object.wait(Native Method)
        - waiting on <0x00000007ad001310> (a java.lang.ref.ReferenceQueue$Lock)
        at java.lang.ref.ReferenceQueue.remove(ReferenceQueue.java:133)
        - locked <0x00000007ad001310> (a java.lang.ref.ReferenceQueue$Lock)
        at java.lang.ref.ReferenceQueue.remove(ReferenceQueue.java:149)
        at java.lang.ref.Finalizer$FinalizerThread.run(Finalizer.java:177)

"Reference Handler" daemon prio=10 tid=0x00007fd7ec07e000 nid=0x66a5 in Object.wait() [0x00007fd7f0353000]
    java.lang.Thread.State: WAITING (on object monitor)
        at java.lang.Object.wait(Native Method)
        - waiting on <0x00000007ad0011e8> (a java.lang.ref.Reference$Lock)
        at java.lang.Object.wait(Object.java:502)
        at java.lang.ref.Reference$ReferenceHandler.run(Reference.java:133)
        - locked <0x00000007ad0011e8> (a java.lang.ref.Reference$Lock)
4

4 回答 4

2

JVM 有许多线程在运行以执行“内务处理”,例如垃圾收集等。这些线程大部分时间都处于睡眠状态。

于 2012-12-30T23:47:50.577 回答
1

然后,运行 top 显示进程状态为 S,它正在使用 100% 的 CPU 时间。

我认为问题很可能与top确定流程状态的方式有关。在操作系统级别,具有运行/等待/睡眠状态的是各个线程。我的猜测是,该top命令用于确定进程的名义运行/等待/睡眠状态的启发式方法具有误导性。

(以及jstack......的输出和常识......支持这个理论。)

于 2012-12-30T23:59:08.063 回答
0

该线程正忙于使用 CPU,它没有休眠。

"main" prio=10 tid=0x00007fd7ec007800 nid=0x669b runnable [0x00007fd7f5754000]
  java.lang.Thread.State: RUNNABLE
    at Test.main(Test.java:5)

这表明顶部的信息不正确。

于 2012-12-30T23:49:05.960 回答
0

有很多关于“为什么你有多个线程,以及它们在 Java 中做什么”的解释,但关于什么的解释不多。该top命令本身相当简单,因为它只是通过收集信息所需的系统调用 - 例如getrusage用于获取 CPU 使用率。

通常,正如所解释的那样,top一次查看整个进程,所有线程。您可以按“H”,top它将显示单个线程而不是整个进程,然后您将能够看到所有 Java 线程 [如果您的屏幕足够高],其中主要线程正在使用您的所有 CPU无限循环,其他线程主要是什么都不做。

于 2012-12-31T08:16:39.293 回答