дождитесь завершения задачи, не блокируя пользовательский интерфейс в javafx - PullRequest
0 голосов
/ 25 октября 2018

Я пытаюсь создать приложение для викторины, используя JavaFX, так как я вызываю вопросы, используя

Q1.invoke();
Q2.invoke();

, эти вопросы будут отображаться в пользовательском интерфейсе

public void display(McqQuestion mcqQuestion) {
        resourceAsStream  = getClass().getResourceAsStream("/mcqview.fxml");
        fxmlLoader = new FXMLLoader();
        if (executorService==null) executorService =Executors.newSingleThreadExecutor();
        Parent root = null;
        try {
            root = fxmlLoader.load(resourceAsStream);
            Mcqview controller = fxmlLoader.getController();
            controller.setAnswer1(mcqQuestion.getAnswers().get(0));
            //controller class has setters to accept question properties.
            controller.multipleChoiceQuestionType = this;
            this.view.getBorderPane().setCenter(root);
}

после того, как вопросМне нужно подождать, пока я не получу ответ. Если я не получил ответа, должен быть вызван следующий вопрос. Так что я ввел нить в метод отображения для ожидания тайм-аута

submit = executorService.submit(() -> {
             try {
                    TimeUnit.SECONDS.sleep(2000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
             });

            try {
                submit.get(20,TimeUnit.SECONDS);
                System.out.println("waiting finished");
            } catch (InterruptedException e) {
                e.printStackTrace();
            }

поскольку future.get(); является блокирующим вызовом, он также блокирует поток пользовательского интерфейса, как этого добиться, не блокируя поток пользовательского интерфейса.

Ответы [ 2 ]

0 голосов
/ 25 октября 2018

Не используйте отдельный поток для этой цели.Это только усложняет ситуацию.JavaFX предоставляет способы ожидания, которые не требуют от вас проблем с параллелизмом.

В этом случае ожидание можно выполнить из PauseTransition с обработчиком onFinished.Обработайте ответ из обработчика событий для пользовательского ввода.

private static class Question {

    private final String questionText;
    private final String answers[];
    private final int correctAnswerIndex;

    public Question(String questionText, String[] answers, int correctAnswerIndex) {
        if (answers.length != 3) {
            // for simplicity's sake allow only exactly 3 answers
            throw new IllegalArgumentException();
        }
        this.questionText = questionText;
        this.answers = answers;
        this.correctAnswerIndex = correctAnswerIndex;
    }

}

private VBox questionPane;
private Label questionText;
private Button[] answerButtons;
private PauseTransition pauseTransition;
private Question currentQuestion;

private void answer(int index) {
    pauseTransition.stop(); // no longer wait for timeout
    Alert alert = new Alert(Alert.AlertType.INFORMATION);
    alert.setContentText((index == currentQuestion.correctAnswerIndex)
            ? "correct answer"
            : "incorrect answer");

    // show result and exit
    alert.showAndWait();
    Platform.exit();
}

private void ask(Question question) {
    questionText.setText(question.questionText);
    for (int i = 0; i < 3; i++) {
        answerButtons[i].setText(question.answers[i]);
    }
    currentQuestion = question;
    pauseTransition.playFromStart(); // start timeout timer
}

private void timeout() {
    pauseTransition.stop();
    Alert alert = new Alert(Alert.AlertType.INFORMATION);
    alert.setContentText("your time ran out");

    // cannot use showAndWait form animation directly
    Platform.runLater(() -> {
        // show result and exit
        alert.showAndWait();
        Platform.exit();
    });
}

@Override
public void start(Stage stage) {
    pauseTransition = new PauseTransition(Duration.seconds(10));
    pauseTransition.setOnFinished(evt -> timeout());

    questionText = new Label();
    questionText.setWrapText(true);

    questionPane = new VBox(questionText);
    questionPane.setPrefSize(400, 400);
    answerButtons = new Button[3];

    for (int i = 0; i < 3; i++) {
        final int answerIndex = i;
        Button button = new Button();
        button.setOnAction(evt -> answer(answerIndex));
        answerButtons[i] = button;
        questionPane.getChildren().add(button);
    }

    Scene scene = new Scene(questionPane);

    stage.setScene(scene);
    stage.show();

    Question question = new Question(
            "What is the answer to the ultimate question of life, the universe, and everything?",
            new String[]{"Mew", "42", "Peanut butter"},
            1
    );
    ask(question);
}

Вы можете легко реализовать время ожидания или результат ответа на вопрос другим способом, например, задав следующий вопрос или показывая результаты, когдапоследний вопрос сделан.

0 голосов
/ 25 октября 2018

Для изменений в пользовательском интерфейсе вы должны использовать

Platform.runLater(() -> {

});

, а для Thread вы должны использовать:

Task<Void> task = new Task<Void>() {
    @Override
    protected Void call() throws Exception {
                return null;
        }
};

и передавать объект задачи в

executorService.submit(task)

hopeэто будет полезно

...