8

我正在尝试在应用程序 GUI 中模拟基本恒温器。

我想用新的温度值每 2 秒更新一次标签框值。

例如,我的初始温度将显示为 68 度,然后每 2 秒更新到 69、70 等,直到 75。

这是我用 Java fx 编写的一段代码。 controlpanel是存在标签框的形式的对象。它仅将最终值更新为 75。它不会每 2 秒更新一次。我写了一个方法 pause 导致 2 秒延迟。所有标签都用它们的最终值更新,但不是每 2 秒更新一次。当我调试时,我可以看到这些值每 2 秒增加一个。这段代码写在按钮 onClick 事件中

private void jButton1ActionPerformed(java.awt.event.ActionEvent evt) {                                         
    int i=0;
    Timer asd = new Timer(1000,null);

    asd.setDelay(1000);

    while(i < 10)
    {
         jTextField1.setText(Integer.toString(i));
         i++;

         asd.start();
    }
 }  
4

3 回答 3

17

要使用 Timer 解决您的任务,您需要TimerTask使用您的代码实现并使用Timer#scheduleAtFixedRate方法重复运行该代码:

Timer timer = new Timer();
    timer.scheduleAtFixedRate(new TimerTask() {
        @Override
        public void run() {
            System.out.print("I would be called every 2 seconds");
        }
    }, 0, 2000);

另请注意,调用任何 UI 操作必须在 Swing UI 线程(或 FX UI 线程,如果您使用 JavaFX)上完成:

private int i = 0;
private void jButton1ActionPerformed(java.awt.event.ActionEvent evt) {
    Timer timer = new Timer();
    timer.scheduleAtFixedRate(new TimerTask() {
        @Override
        public void run() {
            SwingUtilities.invokeLater(new Runnable() {
                @Override
                public void run() {
                    jTextField1.setText(Integer.toString(i++));
                }
            });
        }
    }, 0, 2000);
}

在 JavaFX 的情况下,您需要在“FX UI 线程”而不是 Swing 上更新 FX 控件。实现该使用javafx.application.Platform#runLater方法而不是SwingUtilities

于 2013-04-21T23:39:33.000 回答
15

这是使用 JavaFX 动画时间轴而不是计时器的替代解决方案。

我喜欢这个解决方案,因为动画框架确保一切都发生在 JavaFX 应用程序线程上,因此您无需担心线程问题。

温度探针

import javafx.animation.*;
import javafx.application.Application;
import javafx.beans.binding.Bindings;
import javafx.beans.property.*;
import javafx.event.*;
import javafx.scene.*;
import javafx.scene.control.*;
import javafx.scene.layout.VBox;
import javafx.stage.Stage;
import javafx.util.Duration;

import java.util.Random;

public class ThermostatApp extends Application {
  @Override public void start(final Stage stage) throws Exception {
    final Thermostat       thermostat       = new Thermostat();
    final TemperatureLabel temperatureLabel = new TemperatureLabel(thermostat);

    VBox layout = new VBox(10);
    layout.getChildren().addAll(temperatureLabel);
    layout.setStyle("-fx-background-color: cornsilk; -fx-padding: 20; -fx-font-size: 20;");

    stage.setScene(new Scene(layout));
    stage.show();
  }

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

class TemperatureLabel extends Label {
  public TemperatureLabel(final Thermostat thermostat) {
    textProperty().bind(
      Bindings.format(
        "%3d \u00B0F",
        thermostat.temperatureProperty()
      )
    );
  }
}

class Thermostat {
  private static final Duration PROBE_FREQUENCY = Duration.seconds(2);

  private final ReadOnlyIntegerWrapper temperature;
  private final TemperatureProbe       probe;
  private final Timeline               timeline;

  public ReadOnlyIntegerProperty temperatureProperty() {
    return temperature.getReadOnlyProperty();
  }

  public Thermostat() {
    probe       = new TemperatureProbe();
    temperature = new ReadOnlyIntegerWrapper(probe.readTemperature());

    timeline = new Timeline(
        new KeyFrame(
          Duration.ZERO,
          new EventHandler<ActionEvent>() {
            @Override public void handle(ActionEvent actionEvent) {
              temperature.set(probe.readTemperature());
            }
          }
        ),
        new KeyFrame(
          PROBE_FREQUENCY
        )
    );
    timeline.setCycleCount(Timeline.INDEFINITE);
    timeline.play();
  }
}

class TemperatureProbe {
  private static final Random random = new Random();

  public int readTemperature() {
    return 72 + random.nextInt(6);
  }
}

该解决方案基于以下倒数计时器解决方案:JavaFX:如何绑定两个值?

于 2013-04-22T02:01:03.977 回答
0

通话Platform.runLater对我有用:

Platform.runLater(new Runnable() {

    @Override
    public void run() {

    }
});
于 2018-07-12T18:09:09.357 回答