问题:
- 尽管主
javafx.application.Application
类成功运行并显示GUI窗口,但运行JavaFX应用程序jar失败
我有什么:
- IntelliJ IDEA社区版2021.2.1
- Liberica完整JDK 11.0.12+7 x86 64位Windows版(已包含JavaFX)
- Windows 10 Home 64位
项目结构(IDE生成,未修改):
main
├───java
│ │ module-info.java
│ │
│ ├───com
│ │ └───tejasb
│ │ └───test
│ │ HelloApplication.java
│ │ HelloController.java
│ │
│ └───META-INF
│ MANIFEST.MF
│
└───resources
└───com
└───tejasb
└───test
hello-view.fxml
源代码:
- HelloApplication.java
package com.tejasb.test;
import javafx.application.Application;
import javafx.fxml.FXMLLoader;
import javafx.scene.Scene;
import javafx.stage.Stage;
import java.io.IOException;
public class HelloApplication extends Application {
@Override
public void start(Stage stage) throws IOException {
FXMLLoader fxmlLoader = new FXMLLoader(HelloApplication.class.getResource("hello-view.fxml"));
Scene scene = new Scene(fxmlLoader.load(), 320, 240);
stage.setTitle("Hello!");
stage.setScene(scene);
stage.show();
}
public static void main(String[] args) {
launch();
}
}
- HelloController.java
package com.tejasb.test;
import javafx.fxml.FXML;
import javafx.scene.control.Label;
public class HelloController {
@FXML
private Label welcomeText;
@FXML
protected void onHelloButtonClick() {
welcomeText.setText("Welcome to JavaFX Application!");
}
}
- hello-view.fxml
<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.geometry.Insets?>
<?import javafx.scene.control.Label?>
<?import javafx.scene.layout.VBox?>
<?import javafx.scene.control.Button?>
<VBox alignment="CENTER" spacing="20.0" xmlns:fx="http://javafx.com/fxml"
fx:controller="com.tejasb.test.HelloController">
<padding>
<Insets bottom="20.0" left="20.0" right="20.0" top="20.0"/>
</padding>
<Label fx:id="welcomeText"/>
<Button text="Hello!" onAction="#onHelloButtonClick"/>
</VBox>
- 模块-info.java
module com.tejasb.test {
requires javafx.controls;
requires javafx.fxml;
opens com.tejasb.test to javafx.fxml;
exports com.tejasb.test;
}
- 歧管.MF
Manifest-Version: 1.0
Main-Class: com.tejasb.test.HelloApplication
异常堆栈跟踪(当应用程序jar运行时):
Exception in Application start method
java.lang.reflect.InvocationTargetException
at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.base/java.lang.reflect.Method.invoke(Method.java:566)
at javafx.graphics/com.sun.javafx.application.LauncherImpl.launchApplicationWithArgs(LauncherImpl.java:464)
at javafx.graphics/com.sun.javafx.application.LauncherImpl.launchApplication(LauncherImpl.java:363)
at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.base/java.lang.reflect.Method.invoke(Method.java:566)
at java.base/sun.launcher.LauncherHelper$FXHelper.main(LauncherHelper.java:1051)
Caused by: java.lang.RuntimeException: Exception in Application start method
at javafx.graphics/com.sun.javafx.application.LauncherImpl.launchApplication1(LauncherImpl.java:900)
at javafx.graphics/com.sun.javafx.application.LauncherImpl.lambda$launchApplication$2(LauncherImpl.java:195)
at java.base/java.lang.Thread.run(Thread.java:829)
Caused by: java.lang.IllegalStateException: Location is not set.
at javafx.fxml/javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:2459)
at javafx.fxml/javafx.fxml.FXMLLoader.load(FXMLLoader.java:2435)
at com.tejasb.test.HelloApplication.start(HelloApplication.java:14)
at javafx.graphics/com.sun.javafx.application.LauncherImpl.lambda$launchApplication1$9(LauncherImpl.java:846)
at javafx.graphics/com.sun.javafx.application.PlatformImpl.lambda$runAndWait$12(PlatformImpl.java:455)
at javafx.graphics/com.sun.javafx.application.PlatformImpl.lambda$runLater$10(PlatformImpl.java:428)
at java.base/java.security.AccessController.doPrivileged(Native Method)
at javafx.graphics/com.sun.javafx.application.PlatformImpl.lambda$runLater$11(PlatformImpl.java:427)
at javafx.graphics/com.sun.glass.ui.InvokeLaterDispatcher$Future.run(InvokeLaterDispatcher.java:96)
at javafx.graphics/com.sun.glass.ui.win.WinApplication._runLoop(Native Method)
at javafx.graphics/com.sun.glass.ui.win.WinApplication.lambda$runLoop$3(WinApplication.java:174)
... 1 more
Exception running application com.tejasb.test.HelloApplication
应用程序jar的内容:
Test.jar
│ module-info.class
│
├───com
│ └───tejasb
│ └───test
│ HelloApplication.class
│ HelloController.class
│
├───META-INF
│ MANIFEST.MF
│
└───resources
└───com
└───tejasb
└───test
hello-view.fxml
重现问题的步骤:
- 在IntelliJ IDEA中创建项目,项目结构如上
- 将新的运行配置创建为
Application
- 从下拉菜单和主类中将Name、JDK/JJRE、-cp设置为
com.tejasb.test.HelloApplication
- 从下拉菜单和主类中将Name、JDK/JJRE、-cp设置为
- 转到项目结构>项目设置>人工制品
- 单击+,然后单击JAR>来自具有依赖项的模块
- 将Main类设置为
com.tejasb.test.HelloApplication
,然后选择复制到输出目录并通过清单链接 - 单击"确定",然后单击右侧
- 右键单击.jar,选择Create directory并将名称设置为
resources
- 然后右键单击创建的目录并选择添加>目录内容并选择src>main>资源目录(来自项目)
- 单击"确定">
- 单击"运行"按钮(应用程序应成功运行,显示一个包含按钮的窗口)
- 现在,单击"生成">生成项目>build(这应该在您指定的位置构建一个应用程序jar文件)
- 最后,当这个生成的jar从控制台运行时,它不起作用,抛出上面的异常
我尝试过的:
- 设置
FXMLLoader fxmlLoader = new FXMLLoader(HelloApplication.class.getResource("hello-view.fxml"));
到这个
FXMLLoader fxmlLoader = new FXMLLoader(HelloApplication.class.getResource("/hello-view.fxml"));
即使在步骤4(再现步骤的问题)和步骤5(再现步骤问题)中,也会抛出与上述相同的异常。
- 此外,尝试使用
HelloApplication.class.getClassLoader().getResource("hello-view.fxml")
和HelloApplication.class.getClassLoader().getResource("/hello-view.fxml")
以及其他。。。。但是,罐子抛出了同样的异常
如有任何帮助,我们将不胜感激。感谢
创建这个问题花费了大量的工作。
您的问题是,您的jar中不应该有一个名为resources
的目录。
相反,hello-view.fxml
应该与HelloApplication
在jar中的位置相同。
在资源查找中不需要/
。
我不知道如何用当前的构建方法修复资源包。
我建议使用像Maven这样的构建工具,而不是IDE内置的工件创建。我想在openjfx.io上可能会有这样的教程。使用您当前的目录结构,Maven会自动将所有输出文件放在工件jar的正确位置。
如果你真的想在你的jar中保留一个名为resources的文件夹(这是非常不寻常的,不推荐),那么你可以保持原样,并使用路径/resources/com/tejasb/test/hello-view.fxml
查找你的fxml。请注意,询问者在评论中回答说,这种方法不起作用。