У меня есть SwingWorker
, который взаимодействует с сервером в фоновом режиме, а затем обновляет JFrame
. Я отлаживал свое приложение и заметил, что даже после того, как SwingWorker
закончил свою работу, его поток все еще оставался. Он висит на Unsafe.park(java.lang.Object)
, который является нативным методом. Я изучил это дальше и обнаружил, что все мои другие SwingWorker
в моем приложении делают то же самое после завершения. Я могу предоставить исходный код, если кто-то хочет, но я не думаю, что это необходимо, потому что проблема кажется очень общей.
Я запустил приложение без отладчика, и проблема все еще возникает. Это дамп потока SwingWorker
:
"SwingWorker-pool-2-thread-1" daemon prio=6 tid=0x03219800 nid=0xd74 waiting on
condition [0x04b7f000]
java.lang.Thread.State: WAITING (parking)
at sun.misc.Unsafe.park(Native Method)
- parking to wait for <0x22ec63d8> (a java.util.concurrent.locks.Abstra
ctQueuedSynchronizer$ConditionObject)
at java.util.concurrent.locks.LockSupport.park(Unknown Source)
at java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject
.await(Unknown Source)
at java.util.concurrent.LinkedBlockingQueue.take(Unknown Source)
at java.util.concurrent.ThreadPoolExecutor.getTask(Unknown Source)
at java.util.concurrent.ThreadPoolExecutor.runWorker(Unknown Source)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(Unknown Source)
at java.lang.Thread.run(Unknown Source)
Я сделал пример программы, которая использует SwingWorker
так, как она обычно используется в приложении. У этой программы та же проблема. Вот код:
package swingworkerlocktest;
import java.util.List;
import javax.swing.JFrame;
import javax.swing.JTextArea;
import javax.swing.SwingWorker;
public class SwingWorkerLockTest {
public static void main(String[] args) {
final JFrame frame = new JFrame("Frame");
final JTextArea outputArea = new JTextArea(4, 20);
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.add(outputArea);
frame.pack();
frame.setVisible(true);
(new SwingWorker<Object, String>() {
@Override
protected Object doInBackground() throws Exception {
publish("Background task.");
return null;
}
@Override
protected void process(List<String> chunks) {
for (String str : chunks) {
outputArea.append(str + "\n");
}
}
@Override
protected void done() {
outputArea.append("Background task finished.");
}
}).execute();
}
}