我正在调试一些使用SwingWorker编写的代码来执行数值计算和GUI更新的混合。 SwingWorker挂起以下堆栈跟踪:
Full thread dump Java HotSpot(TM) Client VM (14.3-b01 mixed mode, sharing):
"SwingWorker-pool-3-thread-4" prio=6 tid=0x07fd7c00 nid=0x143c waiting on condition [0x0a33f000]
java.lang.Thread.State: TIMED_WAITING (sleeping)
at java.lang.Thread.sleep(Native Method)
at sum.ee.ui.modelviewer.ModelViewer$ModelAnimator.doInBackground(ModelViewer.java:940)
at sum.ee.ui.modelviewer.ModelViewer$ModelAnimator.doInBackground(ModelViewer.java:877)
at javax.swing.SwingWorker$1.call(SwingWorker.java:274)
at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:303)
at java.util.concurrent.FutureTask.run(FutureTask.java:138)
at javax.swing.SwingWorker.run(SwingWorker.java:313)
at java.util.concurrent.ThreadPoolExecutor$Worker.runTask(ThreadPoolExecutor.java:886)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:908)
at java.lang.Thread.run(Thread.java:619)
"SwingWorker-pool-3-thread-3" prio=6 tid=0x07fd7000 nid=0x11a8 waiting for monitor entry [0x0a2af000]
java.lang.Thread.State: BLOCKED (on object monitor)
at java.awt.Component.resize(Component.java:2044)
- waiting to lock <0x24b936a0> (a java.awt.Component$AWTTreeLock)
at java.awt.Component.setSize(Component.java:2035)
at java.awt.Component.resize(Component.java:2069)
at java.awt.Component.setSize(Component.java:2060)
at javax.swing.JViewport.setViewSize(JViewport.java:1038)
at javax.swing.ViewportLayout.layoutContainer(ViewportLayout.java:183)
at java.awt.Container.layout(Container.java:1421)
at java.awt.Container.doLayout(Container.java:1410)
at jsyntaxpane.components.LineNumbersRuler.updateSize(LineNumbersRuler.java:109)
at jsyntaxpane.components.LineNumbersRuler.removeUpdate(LineNumbersRuler.java:203)
at javax.swing.text.AbstractDocument.fireRemoveUpdate(AbstractDocument.java:243)
at jsyntaxpane.SyntaxDocument.fireRemoveUpdate(SyntaxDocument.java:118)
at javax.swing.text.AbstractDocument.handleRemove(AbstractDocument.java:608)
at javax.swing.text.AbstractDocument.remove(AbstractDocument.java:576)
at javax.swing.JEditorPane.setText(JEditorPane.java:1493)
at sum.ee.ui.SourceCodePanel.clearSourcePane(SourceCodePanel.java:256)
at sum.ee.ui.SourceCodePanel.access$100(SourceCodePanel.java:47)
at sum.ee.ui.SourceCodePanel$1.stateChanged(SourceCodePanel.java:209)
at sum.ee.ui.VisualizationAggregator.fireStateChanged(VisualizationAggregator.java:300)
at sum.ee.ui.VisualizationAggregator.update(VisualizationAggregator.java:97)
at sum.ee.ui.modelviewer.ModelViewer$ModelAnimator.doInBackground(ModelViewer.java:918)
at sum.ee.ui.modelviewer.ModelViewer$ModelAnimator.doInBackground(ModelViewer.java:877)
at javax.swing.SwingWorker$1.call(SwingWorker.java:274)
at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:303)
at java.util.concurrent.FutureTask.run(FutureTask.java:138)
at javax.swing.SwingWorker.run(SwingWorker.java:313)
at java.util.concurrent.ThreadPoolExecutor$Worker.runTask(ThreadPoolExecutor.java:886)
at java.util.concurrent.ThreadPoolExecutor
$ Worker.run(ThreadPoolExecutor.java:908) 在java.lang.Thread.run(Thread.java:619)
我的理解是GUI工作不应该在doInBackground()内部完成,而应该在done()中完成。我进行了一个天真的实验,我把doInBackground()中的所有代码都移到了done()中,但仍然没有用。是否有任何提示,人们可以告诉我我可以做什么来根本导致这个问题?代码如下所示:
protected Void doInBackground() {
isAnimating = true;
resetButtonBackgrounds();
backgroundColor = new Color(175, 255, 175); // Soft Green
JToggleButton b = null;
for (final int index : modelIndices) {
if (index == modelIndices.get(modelIndices.size() - 1)) {
backgroundColor = defaultBackgroundColor;
}
if (!keepTrace) {
// Resetting the backgrounds is necessary to have
// individual display of the changing elements due to
// the fact that there can be multiple nodes per
// source line. The reset works in combination
// with updating from ModelViewer.this (as opposed
// to the 'this' of ModelAnimator instances) due
// to not sending an event to itself. Furthermore,
// if the event was sent from ModelAnimator, the model
// indices are recalculated, causing a jump when multiple
// element source lines are encountered.
resetButtonBackgrounds();
}
aggregator.modelIndex(index);
aggregator.update(ModelViewer.this);
b = getButtonByIndex(index);
scrollRectToVisible(b.getBounds());
ModelViewer.this.repaint();
try {
StaticTools.sleepAtLeast(sleepTimeMilliseconds);
} catch (final InterruptedException ex) {
// continue with thread
}
}
isAnimating = false;
if ( b != null) {
Color orig = b.getBackground();
Color blink = Color.PINK;
Color current = orig;
for (int i = 0; i < 100; i++) {
try {
Thread.sleep(100);
} catch (InterruptedException ex) {
}
if (current == orig) {
current = blink;
} else {
current = orig;
}
b.setBackground(current);
ModelViewer.this.repaint();
}
}
return null;
}
另一个线索是有两个SwingWorker线程可以执行。他们可以运行相同的线程吗?
更新:以下是执行SwingWorker的代码:
public final void animate(final long delayBetweenUpdatesMilliseconds, 最终列表modelIndices, final boolean keepTrace, final List propertyChangeListeners){
ModelAnimator modelAnimator =
new ModelAnimator(delayBetweenUpdatesMilliseconds, modelIndices,
keepTrace);
for (final PropertyChangeListener listener : propertyChangeListeners) {
modelAnimator.addPropertyChangeListener(listener);
}
modelAnimator.execute();
}
答案 0 :(得分:1)
这是未能遵守Swing EDT规则的。
SwingWorker的目的是在发生UI事件时执行繁重的非GUI任务,否则会阻止UI,然后在最后更新UI。
所以你要在doInBackground()中实现你的举重;然后一旦完成swing就会在EDT上调用done(),你可以使用get()来检索结果。
这里的问题是你在SwingWorker创建的新线程中进行GUI工作。这可能会导致死锁和并发问题。
这包括创建所谓的GUI对象,即使你已经在EDT上也应该在runnables中
行动如:
b = getButtonByIndex(index);
应该使用InvokeandWait封装在Runnable中。真正修改GUI本身的东西尤其需要在他们自己的runnables中,甚至如果你已经在swing事件发送线程响应按钮按下或更改,你冒着处理对象的风险已经在努力了。
e.g。 swing工作并锁定A让你做工作,让你做B工作试图锁定和工作A