gpt4 book ai didi

java - 如何将 SwingWorker 与 Swing 同步?

转载 作者:行者123 更新时间:2023-11-30 09:32:21 25 4
gpt4 key购买 nike

问题:我希望主应用程序窗口在所有工作线程停止后关闭。这很容易。但我也想确保 SwingWorker 发送的消息将在关闭窗口发送的消息之前得到处理。我这里有这个(不是那么)小示例源文件。

/*****************************************************************************
TestAsyncEvents.java

This example shows that messages from SwingWorker threads get processed
*AFTER* the WINDOW_CLOSED event even if they had been generated
*BEFORE* them.
The question is: HOW DO I MAKE IT RIGHT?
I know the answer, but it doesn't satisfy me. ;)
*****************************************************************************/
import java.util.List;
import java.awt.Dimension;
import java.awt.Toolkit;
import java.awt.event.WindowAdapter;
import java.awt.event.WindowEvent;
import javax.swing.JFrame;
import javax.swing.SwingUtilities;
import javax.swing.SwingWorker;

//////////////////////////////////////////////////////////////////////////////
//
public class TestAsyncEvents extends JFrame
{

/*
* This class is for closing main window
*/
private class Closer extends WindowAdapter
{
@Override public void windowClosing( WindowEvent ev ) { handleWindowClosing(); }
@Override public void windowClosed( WindowEvent ev ) { handleWindowClosed(); }
}

/*
* This class represents worker that gets asked to stop
* and then reports when it actually stops
*/
private class Ticker extends SwingWorker<Boolean,String>
{
private boolean stop_flag = false;
private int counter = 0;
private boolean result;
Ticker() { super(); execute(); }
@Override protected Boolean doInBackground()
{
// body, executed in worker thread
try {
while( !stop_flag )
{
Thread.sleep(2000);
publish(String.format("Tick #%d",++counter));
}
return result=true;
} catch( Exception ex ) {
return result=false;
}
}
@Override protected void process( List<String> chunks )
{
// reports progress, executed in gui thread
for( String chunk: chunks )
System.out.println(String.format("Chunk processed: %s",chunk));
}
@Override protected void done()
{
// reports result, executed in gui thread
System.out.println(String.format("Thread is %s.",isCancelled()?"cancelled":"stopped normally"));
System.out.println(String.format("Result is %s.",Boolean.toString(result)));
//postClosing(); // IT IS THE SOLUTION FOR MY PROBLEM! BUT... IT ISN'T GOOD ONE!
}
public void askToStop() { stop_flag = true; }
}

/****************************************************************************/
/* FIELDS */
/****************************************************************************/
private static TestAsyncEvents self;
private Ticker worker_object = null;

/****************************************************************************/
/* CONSTRUCTOR */
/****************************************************************************/
TestAsyncEvents()
{
super("Testing Async Events");
addWindowListener(new Closer());
setMinimumSize(new Dimension(512,384));
setVisible(true);
worker_object = new Ticker();
}

/****************************************************************************/
/* INNER METHODS */
/****************************************************************************/
/*
* Waiting for worker to finish
*/
private void doStopping()
{
worker_object.askToStop();
while( !worker_object.isDone() );
}
private boolean stopInSeparateThread()
{
try {
Thread closer = new Thread(new Runnable(){public void run(){doStopping();}});
closer.start();
closer.join();
return true;
} catch( Exception ex ) {
return false;
}
}
private boolean stopHere()
{
doStopping();
return true;
}
private boolean stopWorker()
{
//return stopInSeparateThread();
return stopHere();
}
private boolean canClose()
{
return worker_object.isDone();
}
/*
* Posting WM_CLOSE events
*/
private void doPostCloseEvent()
{
Toolkit.getDefaultToolkit().getSystemEventQueue().postEvent(new WindowEvent(this,WindowEvent.WINDOW_CLOSING));
}
private void postInSeparateThread()
{
SwingUtilities.invokeLater(new Runnable(){public void run(){doPostCloseEvent();}});
}
private void postHere()
{
doPostCloseEvent();
}
private void postClosing()
{
//postInSeparateThread();
postHere();
}
/*
* Methods for Closer class
*/
private void handleWindowClosing()
{
System.out.println("Closing main window...");
if( canClose() )
{
System.out.println("Can close! Disposing...");
dispose();
} else {
System.out.println("Can't close! Now we'll allow it by stopping worker thread...");
boolean res = stopWorker();
System.out.println(String.format("Stopping worker thread went %s.",res?"okay":"wrong"));
postClosing(); // HERE I SIGNAL THE MAIN WINDOW TO CLOSE
}
}
private void handleWindowClosed()
{
System.out.println("Main window is closed!");
}

/****************************************************************************/
/* ENTRY POINT */
/****************************************************************************/
public static void main( final String[] args )
{
SwingUtilities.invokeLater(new Runnable(){public void run(){self=new TestAsyncEvents();}});
System.out.println("All systems are go!");
}

}
//////////////////////////////////////////////////////////////////////////////

它的输出在这里:

F:\C\Java\Test-Frame-Events>java TestAsyncEvents
All systems are go!
Chunk processed: Tick #1
Closing main window...
Can't close! Now we'll allow it by stopping worker thread...
Stopping worker thread went okay.
Closing main window...
Can close! Disposing...
Main window is closed!
Chunk processed: Tick #2
Thread is stopped normally.
Result is true.

我要的是这里:

F:\C\Java\Test-Frame-Events>java TestAsyncEvents
All systems are go!
Chunk processed: Tick #1
Closing main window...
Can't close! Now we'll allow it by stopping worker thread...
Stopping worker thread went okay.
Chunk processed: Tick #2
Thread is stopped normally.
Result is true.
Closing main window...
Can close! Disposing...
Main window is closed!

我突然想到,来自 SwingWorker 的事件是在一个完全不同的事件队列中处理的,而不是处理窗口消息等的事件队列。在我发布 WINDOW_CLOSING 事件之前,我故意等待工作线程停止并发布它的所有消息。但这没有帮助 - 来自 SwingWorker 的消息仍然在 WINDOW_CLOSING 和 WINDOW_CLOSED 事件之后得到处理。这会导致许多微小的不便。特别是,当我关闭 WINDOW_CLOSED 处理程序中的所有日志记录时,希望这些将是我程序中最后执行的运算符,来自工作线程的所有消息在时间和空间上都丢失了。

我知道我的问题的解决方案。我必须取消注释行 #68 和注释行 #161。但这意味着如果我有多个 SwingWorker 线程,我应该生成另一个线程来照顾所有工作人员,并在所有工作人员停止时向主窗口发出退出信号。恕我直言,这并不整洁。那么,Java 专家们,您建议我如何解决这个问题?

最佳答案

一种可能的解决方案:考虑使用 PropertyChangeListener:

private void handleWindowClosing() {
System.out.println("Closing main window...");
if (canClose()) {
System.out.println("Can close! Disposing...");
dispose();
} else {
System.out
.println("Can't close! Now we'll allow it by stopping worker thread...");

worker_object.addPropertyChangeListener(new PropertyChangeListener() {

@Override
public void propertyChange(PropertyChangeEvent pcEvt) {
if (SwingWorker.StateValue.DONE.equals(pcEvt.getNewValue())) {
postClosing();
}
}
});


boolean res = stopWorker();
System.out.println(String.format("Stopping worker thread went %s.",
res ? "okay" : "wrong")); // !!
// postClosing(); // HERE I SIGNAL THE MAIN WINDOW TO CLOSE
}
}

如果你有多个 worker ,你可以使用 CountDownLatchCyclicBarrier

关于java - 如何将 SwingWorker 与 Swing 同步?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12545098/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com