Load()在Java 9之前的风格中工作得很好,但在模块中失败

2022-08-14 00:00:00 java fxml fxmlloader java-module javafx-11

我成功地制作了一个简单的JavaFX FXML程序(在Java 9之前的风格,而不是模块) 我使用了3个文件和主类,如下所示:

└───javafxfxmldemo
        FXMLDocument.fxml
        FXMLDocumentController.java
        JavaFXFXMLDemo.java

主类

public class JavaFXFXMLDemo extends Application {
    @Override
    public void start(Stage stage) throws Exception {
        Parent root = FXMLLoader.load(getClass().getResource("FXMLDocument.fxml"));
        Scene scene = new Scene(root);
        stage.setScene(scene);
        stage.show();
    }

    public static void main(String[] args) {
        launch(args);
    }
}

它在NetBeans 9和DOS终端的javac和Java上都能完美地工作。

现在,我尝试在Java模块中编写相同的程序。以下是我的源代码结构:

└───src
    └───com.fxmlapp
        │   module-info.java
        │
        └───com
            └───fxmlapp
                    FXEventModuleDemo.java
                    FXMLDocument.fxml
                    FXMLDocumentController.java

MODULE-info.java

module com.fxmlapp {
    requires javafx.controls;
    requires javafx.fxml;

    exports com.fxmlapp;
}

除主类名称外,其他所有代码都相同。

它在NetBeans 9上编译,我也可以在DOS终端上编译,如下所示:

javac -d modscom.fxmlapp --module-path %PATH_TO_FX% srccom.fxmlappmodule-info.java srccom.fxmlappcomfxmlappFXEventModuleDemo.java srccom.fxmlappcomfxmlappFXMLDocumentController.java

编译后,我将FXMLDocument.fxml文件放在mods目录中,如下所示。

├───mods
│   └───com.fxmlapp
│       │   module-info.class
│       │
│       └───com
│           └───fxmlapp
│                   FXEventModuleDemo.class
│                   FXMLDocument.fxml
│                   FXMLDocumentController.class
│
└───src
    └───com.fxmlapp
        │   module-info.java
        │
        └───com
            └───fxmlapp
                    FXEventModuleDemo.java
                    FXMLDocument.fxml
                    FXMLDocumentController.java

但是,它无法运行。我在NetBeans 9和DOS终端上都试过了,如下所示:

java --module-path %PATH_TO_FX%;mods -m com.fxmlapp/com.fxmlapp.FXEventModuleDemo

错误来自以下语句:

Parent root = FXMLLoader.load(getClass().getResource("FXMLDocument.fxml"));

我知道,有很多帖子都有这个问题,但当Java模块出现错误时,我找不到解决方案。我认为这不是文件位置问题,因为下面的语句有效。

System.out.println(getClass().getResource("FXMLDocument.fxml"));

FXMLLoader.load()似乎在模块中不起作用。

如果它在非模块样式下工作,我相信它也一定在模块中工作。

实际的错误消息如下所示;是的,我看到了很多关于这个错误的问题,但我仍然找不到Java模块的解决方案)我的模块-info.java或者我的编译或运行命令有问题吗?

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:834)
Caused by: javafx.fxml.LoadException:
/D:/OpenJFX_Demo/FXEventModuleDemo/mods/com.fxmlapp/com/fxmlapp/FXMLDocument.fxml:14

        at javafx.fxml/javafx.fxml.FXMLLoader.constructLoadException(FXMLLoader.java:2625)
        at javafx.fxml/javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:2603)
        at javafx.fxml/javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:2466)
        at javafx.fxml/javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:3237)
        at javafx.fxml/javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:3194)
        at javafx.fxml/javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:3163)
        at javafx.fxml/javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:3136)
        at javafx.fxml/javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:3113)
        at javafx.fxml/javafx.fxml.FXMLLoader.load(FXMLLoader.java:3106)
        at com.fxmlapp/com.fxmlapp.FXEventModuleDemo.start(FXEventModuleDemo.java:13)
        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
Caused by: java.lang.reflect.InaccessibleObjectException: Unable to make field private javafx.scene.control.Label com.fxmlapp.FXMLDocumentController.label accessible: module com.fxmlapp does not "opens com.fxmlapp" to module javafx.fxml
        at java.base/java.lang.reflect.AccessibleObject.checkCanSetAccessible(AccessibleObject.java:340)
        at java.base/java.lang.reflect.AccessibleObject.checkCanSetAccessible(AccessibleObject.java:280)
        at java.base/java.lang.reflect.Field.checkCanSetAccessible(Field.java:176)
        at java.base/java.lang.reflect.Field.setAccessible(Field.java:170)
        at javafx.fxml/javafx.fxml.FXMLLoader$ControllerAccessor.addAccessibleFields(FXMLLoader.java:3479)
        at javafx.fxml/javafx.fxml.FXMLLoader$ControllerAccessor.access$3900(FXMLLoader.java:3328)
        at javafx.fxml/javafx.fxml.FXMLLoader$ControllerAccessor$1.run(FXMLLoader.java:3444)
        at javafx.fxml/javafx.fxml.FXMLLoader$ControllerAccessor$1.run(FXMLLoader.java:3440)
        at java.base/java.security.AccessController.doPrivileged(Native Method)
        at javafx.fxml/javafx.fxml.FXMLLoader$ControllerAccessor.addAccessibleMembers(FXMLLoader.java:3439)
        at javafx.fxml/javafx.fxml.FXMLLoader$ControllerAccessor.getControllerFields(FXMLLoader.java:3378)
        at javafx.fxml/javafx.fxml.FXMLLoader.injectFields(FXMLLoader.java:1170)
        at javafx.fxml/javafx.fxml.FXMLLoader.access$1600(FXMLLoader.java:105)
        at javafx.fxml/javafx.fxml.FXMLLoader$ValueElement.processValue(FXMLLoader.java:865)
        at javafx.fxml/javafx.fxml.FXMLLoader$ValueElement.processStartElement(FXMLLoader.java:759)
        at javafx.fxml/javafx.fxml.FXMLLoader.processStartElement(FXMLLoader.java:2722)
        at javafx.fxml/javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:2552)
        ... 17 more
Exception running application com.fxmlapp.FXEventModuleDemo

解决方案

如@空指针所述:

错误说明模块com.fxmlapp未将com.fxmlapp"打开"到模块javafx.fxml。

因此,我添加了--Add-Open com.fxmlapp/com.fxmlapp=javafx.fxml。它解决了这个问题。

相关文章