Javafx:如果服务消息更新(标签绑定到消息),标签变为空白

Javafx: Label turns blank if service message updates(Label binding to msg)

我是 java 和 javaFx 的新手,我正在尝试处理一个需要在标签上显示一些实时传入数据的项目。

我将我的标签绑定到服务对象的消息,该服务对象不断使用传入数据更新其消息。但是,消息正在更新,但标签变为空白。

没有弹出任何错误,也没有捕获到异常。任何人都可以指出是什么使标签空白而不是与 service.message 一起更新以及如何修复它? 提前致谢。

下面是我正在尝试做的一个简化示例,其中数据源被随机数列表替换。

controller class:我将 scheduledservice 对象放入其中,并将其绑定到标签。

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();

    }
}

Streaming class 包含一个列表,我在其中生成 1000 个随机整数。 processData 方法将 return 并删除列表中的第一个元素。

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;
    }
} 

主要class,展示初级阶段

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>

运行 结果: 这是服务开始前和服务开始后的两张图片。

点击按钮前标签显示 "loading"

单击按钮后标签不显示任何内容

这是程序的部分终端结果。

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

我也是这个社区的新人,如果我在post中做错了什么或者违反了这里的任何规则,也请指出。非常感谢!

ScheduledService 成功时,它会重新安排下一个执行周期。此过程的一部分是 "resetting" 某些属性为其默认值,包括将 message 属性 设置为 ""1。所以发生的事情是您的服务成功,重新启动,并且消息设置回 "" 太快以至于无法在 Label.

中呈现

由于您的代码除了更新消息外什么都不做,因此一种解决方案是 return streaming.processData() 的结果。然后,您将绑定到 ScheduledServicelastValue 属性。它必须是 lastValue 而不是 value 因为后者也会在 reset/restart.

上被清除 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 扩展 Service)。即使 reset() 没有清除属性,启动服务的过程也包括将属性绑定到底层 Task——这是新创建的,没有设置任何属性。

2。此行为是 documented.

添加到

使用 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() 很长并且您想将其保留在后台线程中,您可以像这样更新 gui:

    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();