我的 IDE 是 IntelliJ。我试过this学习预加载器的文档,但由于某种原因,预加载器永远不会从我的主类中调用,甚至它的方法也不会被调用。
这是我的主要类(class):
public class LongInitApp extends Application {
Stage stage;
BooleanProperty ready = new SimpleBooleanProperty(false);
private void longStart() {
//simulate long init in background
Task task = new Task<Void>() {
@Override
protected Void call() throws Exception {
int max = 10;
for (int i = 1; i <= max; i++) {
Thread.sleep(200);
notifyPreloader(new ProgressNotification(((double) i)/max));
}
ready.setValue(Boolean.TRUE);
notifyPreloader(new StateChangeNotification(
StateChangeNotification.Type.BEFORE_START));
return null;
}};
new Thread(task).start();
}
public static void main(String args[]) {
launch(args);
}
@Override
public void start(final Stage stage) throws Exception {
longStart();
stage.setScene(new Scene(new Label("Application started"),400, 400));
ready.addListener(new ChangeListener<Boolean>(){
public void changed(
ObservableValue<? extends Boolean> ov, Boolean t, Boolean t1) {
if (Boolean.TRUE.equals(t1)) {
Platform.runLater(new Runnable() {
public void run() {
stage.show();
}});}}});;}}
还有我的预加载器类:
public class LongAppInitPreloader extends Preloader {
ProgressBar bar;
Stage stage;
boolean noLoadingProgress = true;
private Scene createPreloaderScene() {
bar = new ProgressBar(0);
BorderPane p = new BorderPane();
p.setCenter(bar);
return new Scene(p, 300, 150);
}
public void start(Stage stage) throws Exception {
this.stage = stage;
stage.setScene(createPreloaderScene());
stage.show();
}
@Override
public void handleProgressNotification(ProgressNotification pn) {
if (pn.getProgress() != 1.0 || !noLoadingProgress) {
bar.setProgress(pn.getProgress()/2);
if (pn.getProgress() > 0) {
noLoadingProgress = false;
}
}
}
@Override
public void handleStateChangeNotification(StateChangeNotification evt) {
//ignore, hide after application signals it is ready
}
@Override
public void handleApplicationNotification(PreloaderNotification pn) {
if (pn instanceof ProgressNotification) {
double v = ((ProgressNotification) pn).getProgress();
if (!noLoadingProgress) {
v = 0.5 + v/2;
}
bar.setProgress(v);
} else if (pn instanceof StateChangeNotification) {
stage.hide();
}
}
}
您可以查看这个http://docs.oracle.com/javafx/2/deployment/preloaders.htm文档也是。此代码属于示例 9-11 和 9-12。
最佳答案
您需要在应用程序启动时为其指定预加载器类。实现此目的的一种“快速而肮脏”的方法是使用非公共(public) API 类 com.sun.javafx.application.LauncherImpl。请注意,不保证此类在 JavaFX 的 future 版本中可用,因此您应该仅将其用于快速测试(如果有的话)。在 LongInitApp
类中使用以下 main 方法:
public static void main(String[] args) {
LauncherImpl.launchApplication(LongInitApp.class, LongAppInitPreloader.class, args);
}
包含预加载器的“官方”方法是将其指定为 JavaFX 部署过程的一部分。部署过程的完整文档位于 http://docs.oracle.com/javase/8/docs/technotes/guides/deploy/ ,但最低限度的方法如下。
- 编译应用程序(您的 IDE 通常会在您保存时执行此操作)
从命令行,使用
createjar
命令运行javapackager
工具:javapackager -createjar -outfile myapp.jar -appclass my.package.LongInitApp \ -preloader my.package.LongAppInitPreloader -srcdir dir
其中
my.package
是包含应用程序和预加载器类的包(这些可以位于不同的包中),dir
是包含所有应用程序和预加载器类的目录结构的根目录。类(例如,如果my.package
确实是您的包名称,那么dir
将有一个子目录my
,其中将有一个子目录package
,其中包含.class
文件)。这将生成一个
myapp.jar
文件,该文件是可执行的并且可以识别您的预加载器,因此您可以使用java -jar myapp.jar
执行它。如果您有兴趣,可以从 jar 文件中提取生成的 list ,并使用jar xf myapp.jar META-INF/MANIFEST.MF
查看其中的内容(然后查看文件META-INF/MANIFEST.MF
)。 (简而言之,它的作用是将主类声明为旨在启动 JavaFX 应用程序的内部类。 list 文件包含指定 JavaFX 应用程序类(在您的情况下为LongInitApp
)和预加载器的属性类(如果存在)。启动应用程序的内部类检索这些属性,其 main 方法基本上启动您定义的应用程序,并使用预加载器(如果存在)。)请注意,如果您使用此方法,您的应用程序类
LongInitApp
不需要(并且可能不应该有)一个main
方法。
大多数 IDE 都对此提供某种形式的支持。例如。如果您将 Eclipse 与 E(fx)clipse 插件一起使用,并创建一个 JavaFX 项目,它将为您生成一个 build.fxbuild
文件。双击该文件将打开一个可视化编辑器,您可以在其中设置上面 javapackager
命令中定义的属性。单击“生成 ant build.xml 并运行”将创建 jar 文件。
除非您需要预加载器提供的特定功能(超出您自己可以轻松编程的功能),否则可能不值得为此付出努力。具体来说,如果您通过 Java Web Start 部署应用程序,并且主 jar 文件的下载将需要很长时间,那么预加载器特别有用。发生这种情况时可以显示预加载器。如果您使用的是独立应用程序(或独立的应用程序包),那么您可以非常轻松地自己创建“预加载器”功能。
例如,除非我使用 Java Web Start 并且主 jar 文件很大,否则我可能会通过使“预加载器”只是一个常规的旧类来重构您的示例代码:
import javafx.scene.Scene;
import javafx.beans.property.DoubleProperty ;
import javafx.scene.control.ProgressBar;
import javafx.scene.layout.BorderPane;
import javafx.stage.Stage;
public class LongAppInitPreloader {
private ProgressBar bar;
private Stage stage;
public LongAppInitPreloader() {
this.stage = new Stage();
stage.setScene(createPreloaderScene());
}
private Scene createPreloaderScene() {
bar = new ProgressBar(0);
BorderPane p = new BorderPane();
p.setCenter(bar);
return new Scene(p, 300, 150);
}
public void show() {
stage.show();
}
public void hide() {
stage.hide();
}
public DoubleProperty progressProperty() {
return bar.progressProperty();
}
}
然后您的应用程序类就可以使用任务并更新其进度:
import javafx.application.Application;
import javafx.concurrent.Task;
import javafx.scene.Scene;
import javafx.scene.control.Label;
import javafx.stage.Stage;
public class LongInitApp extends Application {
@Override
public void start(final Stage stage) throws Exception {
Task<Void> task = createLongStartTask();
stage.setScene(new Scene(new Label("Application started"), 400, 400));
LongAppInitPreloader preloader = new LongAppInitPreloader();
preloader.progressProperty().bind(task.progressProperty());
task.setOnSucceeded(e -> {
stage.show();
preloader.hide();
});
preloader.show();
new Thread(task).start();
}
private Task<Void> createLongStartTask() {
// simulate long init in background
Task<Void> task = new Task<Void>() {
@Override
protected Void call() throws Exception {
int max = 10;
for (int i = 1; i <= max; i++) {
Thread.sleep(200);
updateProgress(i, max);
}
return null;
}
};
return task ;
}
// just here so I can run directly from Eclipse:
public static void main(String args[]) {
launch(args);
}
}
关于JavaFX 预加载器从未从 main 调用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44389373/