包含 FXML 从 .jar 中抛出异常

Include FXML throws Exception from .jar

我正在尝试将一个 fxml 文件包含在另一个文件中并将我的应用程序部署为可运行的 jar。

顶层fxml文件是这样加载的:

URL location = Editor.class.getClassLoader().getResource("view/app.fxml");
FXMLLoader fxmlLoader = new FXMLLoader(location);
Parent root = fxmlLoader.load();
appController = fxmlLoader.getController();

app.fxml,位于 "src/view" 中包含此行:

<fx:include fx:id="console" source="console.fxml" />

运行 这在 eclipse 中将按预期工作,但 运行 导出的 .jar 文件将打印:

Exception in Application start method
Exception in thread "main" 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 org.eclipse.jdt.internal.jarinjarloader.JarRsrcLoader.main(JarRsrcLoader.java:61)
Caused by: java.lang.RuntimeException: Exception in Application start method
        at com.sun.javafx.application.LauncherImpl.launchApplication1(LauncherImpl.java:900)
        at com.sun.javafx.application.LauncherImpl.lambda$launchApplication(LauncherImpl.java:195)
        at java.base/java.lang.Thread.run(Thread.java:834)
Caused by: javafx.fxml.LoadException:
view/app.fxml:92

        at javafx.fxml.FXMLLoader.constructLoadException(FXMLLoader.java:2625)
        at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:2603)
        at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:2466)
        at javafx.fxml.FXMLLoader.load(FXMLLoader.java:2435)
        at de.hsa.dice.editor.Editor.start(Editor.java:49)
        at com.sun.javafx.application.LauncherImpl.lambda$launchApplication1(LauncherImpl.java:846)
        at com.sun.javafx.application.PlatformImpl.lambda$runAndWait(PlatformImpl.java:455)
        at com.sun.javafx.application.PlatformImpl.lambda$runLater(PlatformImpl.java:428)
        at java.base/java.security.AccessController.doPrivileged(Native Method)
        at com.sun.javafx.application.PlatformImpl.lambda$runLater(PlatformImpl.java:427)
        at com.sun.glass.ui.InvokeLaterDispatcher$Future.run(InvokeLaterDispatcher.java:96)
        at com.sun.glass.ui.win.WinApplication._runLoop(Native Method)
        at com.sun.glass.ui.win.WinApplication.lambda$runLoop(WinApplication.java:174)
        ... 1 more
Caused by: java.net.MalformedURLException: Could not open InputStream for URL 'rsrc:console.fxml'
        at org.eclipse.jdt.internal.jarinjarloader.RsrcURLConnection.getInputStream(RsrcURLConnection.java:49)
        at java.base/java.net.URL.openStream(URL.java:1117)
        at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:2465)
        at javafx.fxml.FXMLLoader.access00(FXMLLoader.java:105)
        at javafx.fxml.FXMLLoader$IncludeElement.constructValue(FXMLLoader.java:1154)
        at javafx.fxml.FXMLLoader$ValueElement.processStartElement(FXMLLoader.java:754)
        at javafx.fxml.FXMLLoader.processStartElement(FXMLLoader.java:2722)
        at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:2552)
        ... 12 more

请注意,console.fxml 文件与 app.fxml 在同一个包中(在 IDE 和 .jar 文件中)。 这就是为什么我也尝试了 source=".​​/console.fxml",但没有任何改变。

我找到了这个问题的原因和解决方法。

在为包含的 fxml 文件创建 FXMLLoader 时,javafx.fxml.FXMLLoader class 使用此 URL 构造函数:

public URL(URL context, String spec)

使用 FXMLLoader(location) 构造函数中的位置作为上下文,并使用包含元素中的 "source" 作为规范。

因此,当使用路径 "view/app.fxml" 加载我的根 fxml 文件时,上下文将是我的 IDE 中的 "view" 包,而它将是 class导出的 .jar 中的路径。

我尝试使用

创建应用程序-URL
URL classpath = getClass().getProtectionDomain().getCodeSource().getLocation();
URL location = new URL(classpath, "view/app.fxml");

但是当 FXMLLoader 稍后将其再次插入同一个构造函数时,我的上下文将不会再次使用。

所以我能想到的唯一解决方法是将所有 fxml 文件(至少是那些带有 include 标记的文件)放入源文件夹的根目录中。这样,它将直接位于任何环境中的 class 路径。

要从 Eclipse 和作为可运行的 jar 文件成功启动您的 JavaFX 应用程序,您可以尝试在 app.fxml:

中使用绝对 URL
<fx:include fx:id="console" source="/view/console.fxml" />

这样也可以从 Eclipse 导出的可运行 jar 加载 fxml 文件。

除了 Eclipse 的导出工具外,还有 more ways 可以打包 JavaFX 应用程序,尽管我还没有测试当 fxml 文件包含其他 fxml 文件的 include 时它们的行为方式。