Javafx: метка становится пустой при обновлении служебного сообщения (привязка метки к msg) - PullRequest
0 голосов
/ 09 февраля 2019

Я новичок в java, а также в javaFx, я пытаюсь работать над проектом, который должен отображать некоторые текущие входящие данные на этикетках.

Я привязываю свою метку к сообщению сервисного объекта, который постоянно обновляет свое сообщение входящими данными.Однако сообщение обновляется, но метка становится пустой.

Нет ошибок и исключений.Кто-нибудь может указать, что делает ярлык пустым, а не обновляется вместе с service.message и как это исправить?Заранее спасибо.

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

класс контроллера: куда я помещаю объект scheduleservice ви привязать его к этикетке.

import javafx.concurrent.ScheduledService;
import javafx.concurrent.Task;
import javafx.concurrent.WorkerStateEvent;
import javafx.event.ActionEvent;
import javafx.event.EventHandler;
import javafx.fxml.FXML;
import javafx.scene.control.Button;
import javafx.scene.control.Label;
import javafx.util.Duration;

public class Controller {
    @FXML
    Button startButton;

    @FXML
    Label updateLabel;

    @FXML
    void display(ActionEvent event) throws Exception{
        Steaming steaming = new Steaming();

        ScheduledService<Void> service = new ScheduledService<Void>() {
            protected Task<Void> createTask() {
                return new Task<Void>() {
                    protected Void call() {
                        // Call the method and update the message

                        updateMessage(steaming.processData());
                        return null; // Useful in case you want to return data, else null
                    }
                };
            }

        };
        service.setPeriod(Duration.seconds(1)); //Runs every 1 seconds
        service.setOnSucceeded(new EventHandler<WorkerStateEvent>() {
            @Override
            public void handle(WorkerStateEvent t) {
                System.out.println("Message:" + service.messageProperty());
            }
        });

        updateLabel.textProperty().bind(service.messageProperty());
        service.start();

    }
}

Потоковый класс содержит список, в котором я генерирую 1000 случайных чисел.метод processData вернет и удалит 1-й элемент в списке.

import java.util.ArrayList;
import java.util.Collections;
import java.util.List;

public class Steaming{

    List<Integer> numbers;

    Steaming()
    {
        numbers = new ArrayList<>();
        for (int i = 0; i < 1000; i++) {
            numbers.add(i);
        }

        Collections.shuffle(numbers);
    }

    public String processData (){
        String s = "loading";
        try {
            s = this.numbers.get(0).toString();
            numbers.remove(0);
        } catch (Exception e) {
            e.printStackTrace();
        }
        return s;
    }
} 

Основной класс, покажет начальную стадию

import javafx.application.Application;
import javafx.fxml.FXMLLoader;
import javafx.scene.Parent;
import javafx.scene.Scene;
import javafx.stage.Stage;

public class Main extends Application {

    @Override
    public void start(Stage primaryStage) throws Exception{
        Parent root = FXMLLoader.load(getClass().getResource("/sample.fxml"));
        primaryStage.setTitle("Hello World");
        primaryStage.setScene(new Scene(root, 300, 275));
        primaryStage.show();

    }

    public static void main(String[] args) {
        launch(args);
    }
}

Файл FXML

<?xml version="1.0" encoding="UTF-8"?>

<?import javafx.geometry.*?>
<?import javafx.scene.control.*?>
<?import java.lang.*?>
<?import javafx.scene.layout.*?>
<?import javafx.geometry.Insets?>
<?import javafx.scene.layout.GridPane?>
<?import javafx.scene.control.Button?>
<?import javafx.scene.control.Label?>

<GridPane alignment="center" hgap="10" vgap="10" xmlns="http://javafx.com/javafx/8" xmlns:fx="http://javafx.com/fxml/1" fx:controller="sample.Controller">
   <columnConstraints>
      <ColumnConstraints />
   </columnConstraints>
   <rowConstraints>
      <RowConstraints />
   </rowConstraints>
   <children>
      <HBox prefHeight="100.0" prefWidth="200.0">
         <children>
            <Button fx:id="startButton" mnemonicParsing="false" onAction="#display" text="Button">
               <HBox.margin>
                  <Insets right="5.0" />
               </HBox.margin>
            </Button>
            <Label fx:id="updateLabel" prefHeight="26.0" prefWidth="105.0" text="loading">
               <HBox.margin>
                  <Insets left="10.0" />
               </HBox.margin></Label>
         </children>
      </HBox>
   </children>
</GridPane>

ВыполняетсяРезультаты: Вот два рисунка как для запуска службы, так и после.

Метка показывает «загрузку» до нажатия кнопки

enter image description here

После нажатия кнопки

enter image description here

на этикетке ничего не отображается. Вот часть результата терминала, полученного из программы.

Message:StringProperty [bean: sample.Controller$1@7db1e6b2, name: message, bound, value: 33]
Message:StringProperty [bean: sample.Controller$1@7db1e6b2, name: message, bound, value: 200]
Message:StringProperty [bean: sample.Controller$1@7db1e6b2, name: message, bound, value: 389]
Message:StringProperty [bean: sample.Controller$1@7db1e6b2, name: message, bound, value: 188]
Message:StringProperty [bean: sample.Controller$1@7db1e6b2, name: message, bound, value: 915]
Message:StringProperty [bean: sample.Controller$1@7db1e6b2, name: message, bound, value: 76]
Message:StringProperty [bean: sample.Controller$1@7db1e6b2, name: message, bound, value: 205]
Message:StringProperty [bean: sample.Controller$1@7db1e6b2, name: message, bound, value: 583]
Message:StringProperty [bean: sample.Controller$1@7db1e6b2, name: message, bound, value: 181]
Message:StringProperty [bean: sample.Controller$1@7db1e6b2, name: message, bound, value: 872]

IЯ также новичок в этом сообществе, если есть что-то, что я сделал неправильно в посте или нарушил какое-либо из правил здесь, пожалуйста, укажите на это.Большое спасибо!

Ответы [ 2 ]

0 голосов
/ 09 февраля 2019

Добавление к Ответ Slaw :

Будет ли работать JavaFx инструменты анимации для вас?

    Steaming steaming = new Steaming();
    Timeline timeline = new Timeline(
            new KeyFrame(Duration.millis(1000),
            t -> updateLabel.setText(steaming.processData()))
        );
     timeline.setCycleCount(Timeline.INDEFINITE);
     timeline.play();

Если steaming.processData() длинная и вы хотитесохранить его в фоновом потоке, вы можете обновить графический интерфейс следующим образом:

    Steaming steaming = new Steaming();
    ScheduledService<Void> service = new ScheduledService<>() {
        @Override
        protected Task<Void> createTask() {
            return new Task<>() {
                @Override
                protected Void call() {
                    String text = steaming.processData();
                    Platform.runLater(()->  updateLabel.setText(text));
                    return null;
                }
            };
        }
    };
    service.setPeriod(Duration.seconds(1)); //Runs every 1 seconds
    service.start();
0 голосов
/ 09 февраля 2019

Когда ScheduledService завершается успешно, он перепланирует себя для следующего цикла выполнения.Частью этого процесса является «сброс» некоторых свойств до значений по умолчанию, в том числе установка для свойства message значения "" 1 .Итак, происходит то, что ваша служба работает успешно, перезапускается, и сообщение слишком быстро устанавливается на "", чтобы оно отображалось в Label.

Поскольку ваш код ничего не делает, только обновляет сообщение,Одно из решений - вернуть результат streaming.processData().Затем вы привязываете к lastValue свойству ScheduledService.Это должно быть lastValue вместо value, поскольку последний также очищается 2 при перезагрузке / перезапуске.

ScheduledService<String> service = new ScheduledService<>() {

    @Override
    protected Task<String> createTask() {
        return new Task<>() {

            @Override
            protected String call() throws Exception {
                return streaming.processData();
            }

        };
    }

};

updateLabel.textProperty().bind(service.lastValueProperty());

1.Мне не удалось найти документацию по этому вопросу, но реализация Service.reset() показывает, что это так (ScheduledService extends Service).Даже если reset() не очистил свойства, процесс запуска службы включает в себя привязку свойств к базовому Task - который создан недавно и не имеет ни одного из установленных свойств.

2.Это поведение задокументировано .

...