无法运行JavaFX应用程序jar



问题

  • 尽管主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

重现问题的步骤

  1. 在IntelliJ IDEA中创建项目,项目结构如上
  2. 将新的运行配置创建为Application
    • 从下拉菜单和主类中将Name、JDK/JJRE、-cp设置为com.tejasb.test.HelloApplication
  3. 转到项目结构>项目设置>人工制品
    • 单击+,然后单击JAR>来自具有依赖项的模块
    • 将Main类设置为com.tejasb.test.HelloApplication,然后选择复制到输出目录并通过清单链接
    • 单击"确定",然后单击右侧
    • 右键单击.jar,选择Create directory并将名称设置为resources
    • 然后右键单击创建的目录并选择添加>目录内容并选择src>main>资源目录(来自项目)
    • 单击"确定">
  4. 单击"运行"按钮(应用程序应成功运行,显示一个包含按钮的窗口)
  5. 现在,单击"生成">生成项目>build(这应该在您指定的位置构建一个应用程序jar文件)
  6. 最后,当这个生成的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。请注意,询问者在评论中回答说,这种方法不起作用。

相关内容

  • 没有找到相关文章

最新更新