ThreadPoolAsynchronousRunner

ThreadPoolAsynchronousRunner

我正在查看一个jstack日志,这是我所看到的:


  “ com.mchange.v2.async.ThreadPoolAsynchronousRunner $ PoolThread-#2”#250守护程序prio = 5 os_prio = 0 tid = 0x00007f9de0016000 nid = 0x7e54可运行[0x00007f9d6495a000]
     java.lang.Thread.State:RUNNABLE
      在com.mchange.v2.async.ThreadPoolAsynchronousRunner $ PoolThread.run(ThreadPoolAsynchronousRunner.java:534)
      -锁定(com.mchange.v2.async.ThreadPoolAsynchronousRunner)
  
  “ com.mchange.v2.async.ThreadPoolAsynchronousRunner $ PoolThread-#1”#249守护程序prio = 5 os_prio = 0 tid = 0x00007f9de000c000 nid = 0x7e53等待监视器条目[0x00007f9d649db000]
     java.lang.Thread.State:已阻止(在对象监视器上)
      在java.lang.Object.wait(本地方法)
      -等待(com.mchange.v2.async.ThreadPoolAsynchronousRunner)
      在com.mchange.v2.async.ThreadPoolAsynchronousRunner $ PoolThread.run(ThreadPoolAsynchronousRunner.java:534)
      -锁定(com.mchange.v2.async.ThreadPoolAsynchronousRunner)
  
  “ com.mchange.v2.async.ThreadPoolAsynchronousRunner $ PoolThread-#0”#248守护程序prio = 5 os_prio = 0 tid = 0x00007f9de001a000 nid = 0x7e52等待监视​​器条目[0x00007f9d64a5c000]
     java.lang.Thread.State:已阻止(在对象监视器上)
      在java.lang.Object.wait(本地方法)
      -等待(com.mchange.v2.async.ThreadPoolAsynchronousRunner)
      在com.mchange.v2.async.ThreadPoolAsynchronousRunner $ PoolThread.run(ThreadPoolAsynchronousRunner.java:534)
      -锁定(com.mchange.v2.async.ThreadPoolAsynchronousRunner)


因此,在此日志中,这三个线程中的每个线程都设法获得了相同的锁,并且实际上阻塞了底部的两个线程,以等待相同的锁。

有人可以告诉我这个堆栈日志是什么意思吗?

最佳答案

最后两个线程正等待通过将ThreadPoolAsynchronousRunner实例用作监视器来通知,因此该线程的源将如下所示:

synchronized(asyncRunner) {
    // ...
    asyncRunner.wait();
    // ...
}


调用wait后,asyncRunner上的同步就会被“释放”,即应用程序的其他部分可以输入在该实例上同步的块。在您的特定情况下,这似乎已经发生,并且第一个线程的wait-调用已返回,并且当前正在处理来自该线程的某些数据。您仍会在线程转储中看到多条locked行,以向您显示代码当前位于synchronized块中,但是正如所说的,调用wait时释放了“锁”。

在将并发包添加到JDK之前,为了避免进行昂贵的线程创建,在这里您将其视为线程转储的技术非常普遍。而且您的线程转储看起来像是这种实现。这是一个简单的实现,看起来像是“幕后”:

// class ThreadPoolAsynchronousRunner
private Deque<AsyncMessage> queue;

public synchronized void addAsyncMessage(AsyncMessage msg) {
    queue.add(msg);
    notifyAll();
}

public void start() {
    for (int i = 0; i < 4; i++) {
        PoolThread pt = new PoolThread(this);
        pt.start();
    }
}


如果添加了要处理的新消息,则ThreadPoolAsynchronousRunner``将启动PoolThreads并执行notifyAll

// PoolThread

public PoolThread(ThreadPoolAsynchronousRunner parent) {
    this.parent = parent;
}

public void run() {
    try {
        while (true) {
            AsyncMessage  msg = null;
            synchronized(parent) {
                parent.wait();
                if (!parent.queue.isEmpty()) {
                    msg = queue.removeFirst();
                }
            }
            if (msg != null) {
                processMsg(msg);
            }
        }
    }
    catch(InterruptedException ie) {
        // exit
    }
}


notifyAll将导致所有线程的所有wait方法返回,因此您必须检查父级队列中是否仍然包含数据(有时wait会在没有通知的情况下返回,因此您甚至需要进行此检查如果不使用notifyAll)。如果是这种情况,请启动处理方法。您应该在synchronized -block之外执行此操作,否则异步处理类一次只能处理一条消息(除非这就是您想要的-但是为什么要运行多个PoolThread -instances?)

08-26 02:44