无法在另一个包中加载 FXML (JavaFX)

Can't load FXML in another package (JavaFX)

出于某种原因,当我尝试加载位于不同包中的 FXML 时出现错误:

MainApp.java"

FXMLLoader loader = new FXMLLoader();

            System.out.println("view folder: " + MainApp.class.getResource("view/RootLayout.fxml"));     // returns null
            loader.setLocation(MainApp.class.getResource("view/RootLayout.fxml"));

文件夹结构:

错误信息:

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(LauncherImpl.java:195)
        at java.base/java.lang.Thread.run(Thread.java:834)
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 checkmydigitalfootprint.MainApp.initRootLayout(MainApp.java:73)
        at checkmydigitalfootprint.MainApp.start(MainApp.java:55)
        at javafx.graphics/com.sun.javafx.application.LauncherImpl.lambda$launchApplication1(LauncherImpl.java:846)
        at javafx.graphics/com.sun.javafx.application.PlatformImpl.lambda$runAndWait(PlatformImpl.java:455)
        at javafx.graphics/com.sun.javafx.application.PlatformImpl.lambda$runLater(PlatformImpl.java:428)
        at java.base/java.security.AccessController.doPrivileged(Native Method)
        at javafx.graphics/com.sun.javafx.application.PlatformImpl.lambda$runLater(PlatformImpl.java:427)
        at javafx.graphics/com.sun.glass.ui.InvokeLaterDispatcher$Future.run(InvokeLaterDispatcher.java:96)
Exception running application checkmydigitalfootprint.MainApp

我可以告诉你什么对我有用。首先,FXML 文件应该被视为资源而不是 Java 源文件,因此最好将它们放在自己的目录树中。您的源代码当前位于 /src/main/java 树中,因此您的 FXML 文件应移至 /src/main/resources 树中,最好移至名为 fxml 的子目录中。 (我还有一个名为 i18n 的子目录,其中包含用于定义多种语言文本标签的资源包。)

在路径 /src/main/resources/fxml 下找到您的 FXML 文件后,您应该能够从 JavaFX 控制器加载它们,如下所示:

FXMLLoader loader = new FXMLLoader();
URL fxmlLocation = getClass().getResource("/fxml/main_screen.fxml");
loader.setLocation(fxmlLocation);
loader.setController(mainScreenController);
loader.setResources(ResourceBundle.getBundle("i18n/Text", new Locale("sv", "SE")));
Pane pane = loader.<Pane>load();
Scene scene = new Scene(pane);

(如果您的 FXML 文件的根元素不代表 Pane,那么您需要修改调用 load() 方法的行,并将 Pane 替换为合适的类型。)

请注意,对 getResource(String) 的调用采用以正斜杠开头的路径,表示资源路径根 /src/main/resources/

还要注意,奇怪的是,对 getBundle(String) 的调用 而不是 以正斜杠开头,即使您的目标完全相同 /src/main/resources/ 路径。我不得不承认我无法解释为什么这两种方法需要像这样稍微不同地处理,但是这段代码可以同时加载 "main_screen.fxml" 文件和瑞典语资源包文件 "Text_sv_SE.properties".

遇到这种情况的人要尝试的另一件事是确保您的 fxml 文件不为空。在 scenebuilder 中打开它并拖动一个锚窗格。这对我有用