JavaFX中的多线程挂起了UI

时间:2013-02-03 15:52:03

标签: java multithreading javafx-2 javafx

我有一个简单的JavaFX 2应用程序,带有2个按钮,分别表示“开始”和“停止”。单击开始按钮时,我想创建一个后台线程,它将进行一些处理并在其进行时更新UI(例如进度条)。如果单击停止按钮,我希望线程终止。

我尝试使用javafx.concurrent.Task类来执行此操作,我从文档中收集的类可以正常工作。但是每当我点击“开始”时,UI就会冻结/挂起而不是保持正常。

她是来自主Myprogram extends Application课程的代码,用于显示按钮:

public void start(Stage primaryStage)
{               
    final Button btn = new Button();
    btn.setText("Begin");

    //This is the thread, extending javafx.concurrent.Task :
    final MyProcessor handler = new MyProcessor();
    btn.setOnAction(new EventHandler<ActionEvent>()
    {
        public void handle(ActionEvent event)
        {                
           handler.run(); 
        }
    });

    Button stop = new Button();
    stop.setText("Stop");
    stop.setOnAction(new EventHandler<ActionEvent>()
        {
             public void handle(ActionEvent event)
             {
                handler.cancel();
             }
        }

    );
    // Code for adding the UI controls to the stage here.
}

以下是MyProcessor类的代码:

import javafx.concurrent.Task;
public class MyProcessor extends Task
{   
    @Override
    protected Integer call()
    {
        int i = 0;
        for (String symbol : feed.getSymbols() )
        {
            if ( isCancelled() )
            {
                Logger.log("Stopping!");
                return i;
            }
            i++;
            Logger.log("Doing # " + i);
            //Processing code here which takes 2-3 seconds per iteration to execute
            Logger.log("# " + i + ", DONE! ");            
        }
        return i;
    }
}

非常简单,但只要我点击“开始”按钮,UI就会挂起,但控制台消息会继续显示(Logger.log只是System.out.println

我做错了什么?

1 个答案:

答案 0 :(得分:26)

Task实现Runnable,因此当您调用handler.run();时,实际上在UI线程中运行call方法。这会挂起UI。

您应该通过执行程序或仅通过调用new Thread(handler).start();在后​​台线程中启动任务。

the javadocJavaFX concurrency tutorial中解释(可能不是很清楚)。