JavaFX 11:创建标签时出现 IllegalAccessError

新手上路,请多包涵

这个问题可能与这个问题是同 一个 问题,但看起来那个问题的提问者没有添加足够的信息来获得有用的回复。

我正在尝试使用 JDK 和 JavaFx SDK 版本 11.0.2 运行 JavaFx 应用程序。

此代码完全按预期工作,生成一个空窗口:

 import javafx.application.Application;
import javafx.scene.Scene;
import javafx.scene.layout.StackPane;
import javafx.stage.Stage;

public class Main extends Application {

    public void start(Stage primaryStage) throws Exception {
        StackPane root = new StackPane();
        primaryStage.setScene(new Scene(root, 420, 420));
        primaryStage.show();
    }

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

但是,如果我尝试向 StackPane 添加标签,则会引发异常。

 import ...
import javafx.scene.control.Label;

public class Main extends Application {

    public void start(Stage primaryStage) throws Exception {
        StackPane root = new StackPane();
        root.getChildren().add(new Label("42"));
        primaryStage.setScene(new Scene(root, 420, 420));
        primaryStage.show();
    }

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

它产生的堆栈跟踪看起来像这样(Main 中的第 13 行是创建标签的地方):

 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: java.lang.IllegalAccessError: superclass access check failed: class com.sun.javafx.scene.control.ControlHelper (in unnamed module @0xbbd2743) cannot access class com.sun.javafx.scene.layout.RegionHelper (in module javafx.graphics) because module javafx.graphics does not export com.sun.javafx.scene.layout to unnamed module @0xbbd2743
    at java.base/java.lang.ClassLoader.defineClass1(Native Method)
    at java.base/java.lang.ClassLoader.defineClass(ClassLoader.java:1016)
    at java.base/java.security.SecureClassLoader.defineClass(SecureClassLoader.java:174)
    at java.base/jdk.internal.loader.BuiltinClassLoader.defineClass(BuiltinClassLoader.java:802)
    at java.base/jdk.internal.loader.BuiltinClassLoader.findClassOnClassPathOrNull(BuiltinClassLoader.java:700)
    at java.base/jdk.internal.loader.BuiltinClassLoader.loadClassOrNull(BuiltinClassLoader.java:623)
    at java.base/jdk.internal.loader.BuiltinClassLoader.loadClass(BuiltinClassLoader.java:581)
    at java.base/jdk.internal.loader.ClassLoaders$AppClassLoader.loadClass(ClassLoaders.java:178)
    at java.base/java.lang.ClassLoader.loadClass(ClassLoader.java:521)
    at javafx.scene.control.Control.<clinit>(Control.java:86)
    at sample.Main.start(Main.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.gtk.GtkApplication._runLoop(Native Method)
    at javafx.graphics/com.sun.glass.ui.gtk.GtkApplication.lambda$runLoop$11(GtkApplication.java:277)
    ... 1 more
Exception running application sample.Main

我在 Kubuntu 上运行最新版本的 IntelliJ Idea。我从官方网站下载了 Oracle JDK 和 JavaFX

我把 javafx-sdk-11.0.2 放在目录 /usr/lib/jvm/jdk-11.0.2 在同一个目录中。

在 IntelliJ Idea 中,我相信我已经正确选择了 JDK,并且我添加了 /usr/lib/jvm/javafx-sdk-11.0.2/lib 作为库。

IntelliJ Idea 正在使用此命令启动应用程序(为了便于阅读而拆分):

 /usr/lib/jvm/jdk-11.0.2/bin/java
  -Djava.library.path=/usr/lib/jvm/javafx-sdk-11.0.2/lib
  --add-modules javafx.base,javafx.graphics
  --add-reads javafx.base=ALL-UNNAMED
  --add-reads javafx.graphics=ALL-UNNAMED
  -javaagent:/opt/jetbrains/idea-IU-183.4886.37/lib/idea_rt.jar=36031:/opt/jetbrains/idea-IU-183.4886.37/bin
  -Dfile.encoding=UTF-8
  -classpath
    /home/rm/IdeaProjects/JfxPlayground/out/production/JfxPlayground
    :/usr/lib/jvm/javafx-sdk-11.0.2/lib/src.zip
    :/usr/lib/jvm/javafx-sdk-11.0.2/lib/javafx-swt.jar
    :/usr/lib/jvm/javafx-sdk-11.0.2/lib/javafx.web.jar
    :/usr/lib/jvm/javafx-sdk-11.0.2/lib/javafx.base.jar
    :/usr/lib/jvm/javafx-sdk-11.0.2/lib/javafx.fxml.jar
    :/usr/lib/jvm/javafx-sdk-11.0.2/lib/javafx.media.jar
    :/usr/lib/jvm/javafx-sdk-11.0.2/lib/javafx.swing.jar
    :/usr/lib/jvm/javafx-sdk-11.0.2/lib/javafx.controls.jar
    :/usr/lib/jvm/javafx-sdk-11.0.2/lib/javafx.graphics.jar
  -p
    /usr/lib/jvm/javafx-sdk-11.0.2/lib/javafx.base.jar
    :/usr/lib/jvm/javafx-sdk-11.0.2/lib/javafx.graphics.jar
  sample.Main

我没有修改任何 VM 选项。

我该如何解决这个错误?它是 JavaFx 中的错误吗?

原文由 Reinis Mazeiks 发布,翻译遵循 CC BY-SA 4.0 许可协议

阅读 536
2 个回答

您已经在解释您的问题:

我没有修改任何 VM 选项。

由于 JavaFX 11 不再是 JDK 的一部分,您必须从 此处 使用 JavaFX SDK(正如您正在做的那样),或者使用 Maven/Gradle 从 Maven Central 检索 JavaFX 模块。

然后您需要将 SDK 添加为库,以便 IntelliJ 可以找到 JavaFX 类。

但是一旦你这样做了,并且鉴于 JavaFX jar 是模块,你仍然需要做两件事:

  • 使 JavaFX 模块可用于您的模块路径
  • 定义要添加到项目中的模块

根据您的 IntelliJ 输出,它默认添加 javafx.graphicsjavafx.base

 --add-modules javafx.base,javafx.graphics

-p /usr/lib/jvm/javafx-sdk-11.0.2/lib/javafx.base.jar
:/usr/lib/jvm/javafx-sdk-11.0.2/lib/javafx.graphics.jar

(注意 -p--module-path 相同)

这解释了为什么您的项目在第一种情况下运行,当您没有将控件添加到场景时,只有 StackPane 属于 javafx.graphics 模块,但失败添加属于 javafx.controls 模块Label 时发布的异常。

这已被多次提及:您 需要 为您的项目设置所需的 VM 选项。

首先阅读 https://openjfx.io/openjfx-docs/ 上的文档,包括 IntelliJ 文档,您的 IDE 的非模块化项目部分。并阅读第 4 部分。添加 VM 选项。

虚拟机选项

所以点击 Run -> Edit Configurations,然后添加:

 -p /usr/lib/jvm/javafx-sdk-11.0.2/lib --add-modules javafx.controls

应用,运行,问题就解决了。

原文由 José Pereda 发布,翻译遵循 CC BY-SA 4.0 许可协议

另一个选项可能有用,特别是如果您计划将您的应用程序打包为 fat-jar。

感谢 这个答案

您可以创建另一个 Main 不扩展的类 javafx.application.Application 并从那里调用您的应用程序启动方法,而不是修改 vm 选项。尝试以下操作:

 public class App extends Application
{

    @Override public void start(Stage stage)
    {
        // all the stuff necessary
    }

    // Method called "main" here for experimental purposes.
    // But for production use you may decide to call it, "run" for example.
    // Exact name does not matter.
    public static void main(String[] args)
    {
        App.launch();
    }

}

 // This class must NOT extend Application. It is a wrapper only.
public class Runner
{
    public static void main(String[] args)
    {
        // Starter method from previous class
        App.main(args);
    }
}

尝试使用默认配置运行 App.main()VM options 是一个空字段)会导致错误,而 Runner.main() 执行成功。

为了更深入地了解,您可以比较在运行或调试窗口中找到的 java 运行命令(默认情况下会折叠它们)。

原文由 Boris 发布,翻译遵循 CC BY-SA 4.0 许可协议

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题