如何使用进度条

原文: https://docs.oracle.com/javase/tutorial/uiswing/components/progress.html

有时,程序中运行的任务可能需要一段时间才能完成。用户友好的程序向用户提供一些指示,即任务正在发生,任务可能需要多长时间,以及已经完成了多少工作。指示工作的一种方式,也许是进展的量,是使用动画图像。

另一种指示工作的方法是使用 Cursor 类和Component定义的 setCursor 方法设置等待光标。例如,以下代码使光标结束时显示等待光标container(包括它包含的没有指定光标的任何组件):

  1. container.setCursor(Cursor.getPredefinedCursor(Cursor.WAIT_CURSOR));

要传达任务的完整程度,您可以使用如下所示的进度条:

A typical progress bar

有时您无法立即确定长时间运行任务的长度,或者任务可能长时间保持在相同的完成状态。通过将进度条置于不确定模式,您可以显示没有可测量进度的工作。不确定模式下的进度条显示动画以指示正在进行工作。只要进度条可以显示更有意义的信息,您应该将其切换回默认的确定模式。在 Java 外观中,不确定的进度条看起来像这样:

An indeterminate progress bar

Swing 提供了三个类来帮助您使用进度条:

JProgressBar

A visible component to graphically display how much of a total task has completed. See Using Determinate Progress Bars for information and an example of using a typical progress bar. The section Using Indeterminate Mode tells you how to animate a progress bar to show activity before the task’s scope is known.

ProgressMonitor

Not a visible component. Instead, an instance of this class monitors the progress of a task and pops up a dialog if necessary. See How to Use Progress Monitors for details and an example of using a progress monitor.

ProgressMonitorInputStream

An input stream with an attached progress monitor, which monitors reading from the stream. You use an instance of this stream like any of the other input streams described in Basic I/O. You can get the stream’s progress monitor with a call to getProgressMonitor and configure it as described in How to Use Progress Monitors.

在看到进度条和进度监视器正在运行后,决定是否使用进度条或进度监视器可以帮助您找出适合您的应用程序的进度条。

这是一个小型演示应用程序的图片,它使用进度条来衡量在自己的线程中运行的任务的进度:

A snapshot of ProgressBarDemo, which uses a progress bar


Try this:


以下代码来自 ProgressBarDemo.java ,创建并设置进度条:

  1. //Where member variables are declared:
  2. JProgressBar progressBar;
  3. ...
  4. //Where the GUI is constructed:
  5. progressBar = new JProgressBar(0, task.getLengthOfTask());
  6. progressBar.setValue(0);
  7. progressBar.setStringPainted(true);

创建进度条的构造器设置进度条的最小值和最大值。您也可以使用setMinimumsetMaximum设置这些值。此程序中使用的最小值和最大值为 0 以及任务的长度,这是许多程序和任务的典型值。但是,进度条的最小值和最大值可以是任何值,甚至是负值。代码段还将进度条的当前值设置为 0。

setStringPainted的调用导致进度条在其范围内显示已完成任务百分比的文本指示。默认情况下,进度条显示其getPercentComplete方法返回的值,格式为百分比,例如 33%。或者,您可以通过调用setString将默认值替换为其他字符串。例如,

  1. if (/*...half way done...*/)
  2. progressBar.setString("Half way there!");

当用户单击 Start 时,将创建并执行内部类Task的实例。

  1. public void actionPerformed(ActionEvent evt) {
  2. startButton.setEnabled(false);
  3. setCursor(Cursor.getPredefinedCursor(Cursor.WAIT_CURSOR));
  4. done = false;
  5. task = new Task();
  6. task.addPropertyChangeListener(this);
  7. task.execute();
  8. }

Taskjavax.swing.SwingWorker 的子类。 Task实例为ProgressBarDemo做了三件重要事情:

  1. 该实例在单独的线程中调用doInBackground。这是长时间运行的任务实际执行的地方。使用后台线程而不是事件派发线程可防止用户界面在任务运行时冻结。
  2. 后台任务完成后,实例将在事件派发线程中调用done方法。
  3. 该实例维护一个绑定属性progress,该属性已更新以指示任务的进度。每次progress更改时都会调用propertyChange方法。

有关SwingWorker的更多信息,请参阅 Swing 中并发中的工作线程和 SwingWorker

ProgressBarDemo中的后台任务通过以随机间隔报告随机进度量来模拟真实任务。 propertyChange方法通过更新进度条来响应任务的progress属性的更改:

  1. public void propertyChange(PropertyChangeEvent evt) {
  2. if (!done) {
  3. int progress = task.getProgress();
  4. progressBar.setValue(progress);
  5. taskOutput.append(String.format(
  6. "Completed %d%% of task.\n", progress));
  7. }

后台任务完成后,任务的done方法将重置进度条:

  1. public void done() {
  2. //Tell progress listener to stop updating progress bar.
  3. done = true;
  4. Toolkit.getDefaultToolkit().beep();
  5. startButton.setEnabled(true);
  6. setCursor(null); //turn off the wait cursor
  7. progressBar.setValue(progressBar.getMinimum());
  8. taskOutput.append("Done!\n");
  9. }

请注意,done方法将done字段设置为true,防止propertyChange进一步更新进度条。这是必要的,因为在调用done后可能会发生progress属性的最终更新。

ProgressBarDemo2中设置不确定模式,直到实际进度开始:

  1. public void propertyChange(PropertyChangeEvent evt) {
  2. if (!done) {
  3. int progress = task.getProgress();
  4. if (progress == 0) {
  5. progressBar.setIndeterminate(true);
  6. taskOutput.append("No progress yet\n");
  7. } else {
  8. progressBar.setIndeterminate(false);
  9. progressBar.setString(null);
  10. progressBar.setValue(progress);
  11. taskOutput.append(String.format(
  12. "Completed %d%% of task.\n", progress));
  13. }
  14. }
  15. }

代码中的其他更改与字符串显示有关。显示字符串的进度条可能比没有显示字符串的进度条高,并且,作为演示设计者,我们已经任意决定此进度条仅在其处于默认的确定模式时才显示字符串。但是,我们希望避免在更改模式时进度条更改高度时可能导致的布局丑陋。因此,代码在setStringPainted(true)的调用中离开但添加了对setString("")的调用,因此不会显示任何文本。稍后,当进度条从不确定模式切换到确定模式时,调用setString(null)会使进度条显示其默认字符串。

我们做的一个改变*而不是 _ make 正在从progress事件处理器中删除对progressBar.setValue的调用。该调用不会造成任何伤害,因为不确定的进度条不使用其 value 属性,除非可能在状态字符串中显示它。事实上,尽可能保持进度条的数据是最新的,因为一些外观可能不支持不确定模式。


Try this:

  1. 单击“启动”按钮以使用 Java™Web Start下载 JDK 7 或更高版本)运行 ProgressBar2 演示。或者,要自己编译并运行示例,请参考示例索引Launches the ProgressBar2 Demo example

  2. 开始按钮。请注意,一旦按下按钮,进度条就会开始动画,然后切换回确定模式(如 ProgressBarDemo)。


现在让我们重写 ProgressBarDemo 以使用进度监视器而不是进度条。这是新演示程序 ProgressMonitorDemo 的图片:

A snapshot of ProgressMonitorDemo and a dialog brought up by a progress monitor


Try this:

  1. 单击“启动”按钮以使用 Java™Web Start下载 JDK 7 或更高版本)运行 ProgressMonitor 演示。或者,要自己编译并运行示例,请参考示例索引Launches the ProgressMonitor Demo example

  2. 开始按钮。经过一段时间后,程序会显示一个进度对话框。

  3. 单击 OK 按钮。请注意,即使对话框消失,任务仍会继续。
  4. 开始另一项任务。弹出对话框后,单击取消按钮。对话框消失,任务停止。

无法再次使用进度监视器,因此每次启动新任务时都必须创建一个新进度监视器。每次用户使用 Start 按钮开始新任务时,该程序都会创建一个进度监视器。

这是创建进度监视器的语句:

  1. progressMonitor = new ProgressMonitor(ProgressMonitorDemo.this,
  2. "Running a Long Task",
  3. "", 0, task.getLengthOfTask());

此代码使用ProgressMonitor唯一的构造器来创建监视器并初始化多个参数:

  • 第一个参数将父组件提供给进度监视器弹出的对话框。
  • 第二个参数是一个字符串,用于描述被监视任务的性质。该字符串显示在对话框中。有关此参数的详细信息,请参阅 Progress Monitoring API
  • 第三个参数是另一个提供可更改状态注释的字符串。该示例使用空字符串来指示对话框应为可更改的状态注释创建空间,但该注释最初为空。如果为此参数提供null,则会从对话框中省略该注释。每次progress属性更改时,该示例都会更新注释。它同时更新监视器的当前值:

    1. int progress = task.getProgress(); String message = String.format(“Completedd %%。\ n”,progress); progressMonitor.setNotemessage); progressMonitor.setProgress(进展); taskOutput.appendmessage);
  • 最后两个参数分别为对话框中显示的进度条提供最小值和最大值。

默认情况下,进度监视器会在决定是否弹出对话框之前等待 500 毫秒的最小值。它还等待进度超过最小值。如果计算任务完成时间超过 2000 毫秒,则会显示进度对话框。要调整最短等待时间,请调用setMillisToDecidedToPopup。要调整对话框显示所需的最短进度时间,请调用setMillisToPopup

通过此示例使用进度监视器的简单事实,它添加了使用进度条的程序版本中不存在的功能:用户可以通过单击取消按钮取消任务在对话框上。以下是示例中的代码,用于检查用户是否取消了任务或者任务是否正常退出:

  1. if (progressMonitor.isCanceled() || task.isDone()) {
  2. progressMonitor.close();
  3. Toolkit.getDefaultToolkit().beep();
  4. if (progressMonitor.isCanceled()) {
  5. task.cancel(true);
  6. taskOutput.append("Task canceled.\n");
  7. } else {
  8. taskOutput.append("Task completed.\n");
  9. }
  10. startButton.setEnabled(true);
  11. }

请注意,进度监视器本身不会取消该任务。它提供了 GUI 和 API,使程序可以轻松完成。

如果出现以下情况,请使用进度条

  • 您希望更多地控制进度条的配置。如果您直接使用进度条,可以将其设置为不确定,使其垂直显示,提供一个字符串供其显示,在其上注册更改监听器,并为其提供有界范围模型以控制进度条最小值,最大值和当前值。
  • 程序需要显示其他组件以及进度条。
  • 您需要多个进度条。对于某些任务,您需要监视多个参数。例如,除了已成功安装的文件数量之外,安装程序还可以监视磁盘空间使用情况。
  • 您需要重用进度条。进度条可以重复使用;进度监视器不能。一旦进度监视器决定显示对话框(或不显示),进度监视器就不能再次进行。

如果出现以下情况,请使用进度监视器

  • 您想要一种在对话框中显示进度的简便方法。
  • 正在运行的任务是次要的,用户可能对该任务的进度不感兴趣。进度监视器为用户在任务仍在运行时关闭对话框提供了一种方法。
  • 您想要一种简单的方法来取消任务。进度监视器为用户提供了取消任务的 GUI。您所要做的就是调用进度监视器的isCanceled方法来确定用户是否按下了取消按钮。
  • 您的任务在运行时会定期显示一条短消息。进度监视器对话框提供setNote方法,以便任务可以提供有关其正在执行的操作的更多信息。例如,安装任务可能会报告安装时每个文件的名称。
  • 该任务可能不需要很长时间才能完成。您决定运行任务在什么时候花费足够长的时间来保证让用户了解它。如果任务在您设置的时间范围内完成,则进度监视器不会弹出对话框。

如果您决定使用进度监视器,您正在监视的任务是从输入流中读取,请使用 ProgressMonitorInputStream类。

下表列出了使用进度条和进度监视器的常用 API。因为JProgressBarJComponent的子类,所以您可能在JProgressBar上调用的其他方法列在 JComponent 类中。请注意,ProgressMonitorObject的子类,不是可视组件。

用于监控进度的 API 分为以下几类:

构造器 目的
JProgressBar()

JProgressBar(int,int) | 创建一个水平进度条。无参数构造器初始化进度条,其最小值和初始值为 0,最大值为 100.具有两个整数参数的构造器指定最小值和最大值。 | | JProgressBar(int) JProgressBar(int,int,int) | 创建具有指定方向的进度条,可以是JProgressBar.HORIZONTALJProgressBar.VERTICAL。可选的第二个和第三个参数指定最小值和最大值。 | | JProgressBar(BoundedRangeModel) | 使用指定的范围模型创建水平进度条。 |

方法 目的
void setValue(int)

int getValue() | 设置或获取进度条的当前值。该值受最小值和最大值的约束。 | | double getPercentComplete() | 获取进度条的完成百分比。 | | void setMinimum(int) int getMinimum() | 设置或获取进度条的最小值。 | | void setMaximum(int) int getMaximum() | 设置或获取进度条的最大值。 | | void setModel(BoundedRangeModel) BoundedRangeModel getModel() | 设置或获取进度条使用的模型。该模型建立了进度条的约束和值,因此您可以直接使用它作为使用上面列出的各个 set / get 方法的替代方法。 |

方法 目的
void setIndeterminate(boolean) 通过指定true,将进度条置于不确定模式。指定false会将进度条恢复为默认的确定模式。
void setOrientation(int)

int getOrientation() | 设置或获取进度条是垂直还是水平。可接受的值为JProgressBar.VERTICALJProgressBar.HORIZONTAL。 | | void setBorderPainted(boolean) boolean isBorderPainted() | 设置或获取进度条是否具有边框。 | | void setStringPainted(boolean) boolean isStringPainted() | 设置或获取进度条是否显示百分比字符串。默认情况下,百分比字符串的值是getPercentComplete返回的值,格式为百分比。您可以使用setString设置要显示的字符串。 | | void setString(String) String getString() | 设置或获取百分比字符串。 |

方法或构造器 目的
ProgressMonitor(Component,Object,String,int,int) 创建进度监视器。 Component参数是监视器对话框的父级。 Object参数是放在对话框中选项窗格上的消息。该对象的值通常为StringString参数是一个可更改的状态注释。最后两个int参数分别为对话框中使用的进度条设置最小值和最大值。
ProgressMonitor getProgressMonitor()

(在ProgressMonitorInputStream中) | 获取一个监视来自输入流的读取的进度监视器。 |

方法 目的
void setMinimum(int)

int getMinimum() | 设置或获取进度监视器的最小值。监视器使用此值在对话框中设置进度条。 | | void setMaximum(int) int getMaximum() | 设置或获取进度监视器的最大值。监视器使用此值在对话框中设置进度条。 | | void setProgress(int) | 更新监视器的进度。 | | void setNote(String) String getNote() | 设置或获取状态注释。该注释显示在对话框中。要省略对话框中的状态注释,请提供null作为监视器构造器的第三个参数。 | | void setMillisToDecideToPopup(int) int getMillisToDecideToPopup() | 设置或获取监视器应决定是否弹出对话框的时间。 |

方法 目的
void close() 关闭进度监视器。这将处理对话框。
boolean isCanceled() 确定用户是否按下取消按钮。

以下示例使用JProgressBarProgressMonitor

在哪里描述 笔记
ProgressBarDemo 这个部分 使用基本进度条显示在单独线程中运行的任务的进度。
ProgressBarDemo2 这个部分 使用基本进度条显示在单独线程中运行的任务的进度。
ProgressMonitorDemo 这个部分 修改使用进度监视器而不是进度条的上一个示例。

如果您使用 JavaFX 进行编程,请参阅进度条和进度指示器