JavaFX 11:创建标签时出现 IllegalAccessError
Posted
技术标签:
【中文标题】JavaFX 11:创建标签时出现 IllegalAccessError【英文标题】:JavaFX 11: IllegalAccessError when creating Label 【发布时间】:2019-01-21 14:22:18 【问题描述】:此问题可能与this one 的问题相同,但该问题的提问者似乎没有添加足够的信息来获得有用的回复。
我正在尝试使用 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
我没有修改任何虚拟机选项。
如何解决此错误?它是 JavaFx 中的错误吗?
【问题讨论】:
使用模块运行时,不能使用classpath
声明,反之亦然。
【参考方案1】:
你已经在解释你的问题了:
我没有修改任何虚拟机选项。
由于 JavaFX 11 不再是 JDK 的一部分,因此您必须使用 here 中的 JavaFX SDK(正如您所做的那样),或者使用 Maven/Gradle 从 Maven Central 检索 JavaFX 模块。
然后您需要将 SDK 添加为库,以便 IntelliJ 可以找到 JavaFX 类。
但是一旦你这样做了,并且鉴于 JavaFX jar 是模块,你仍然需要做两件事:
使 JavaFX 模块可用于您的模块路径 定义将哪些模块添加到项目中根据您的 IntelliJ 输出,默认添加 javafx.graphics
和 javafx.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
相同)
这解释了为什么您的项目在第一种情况下运行,当您没有将控件添加到场景中时,只是属于 javafx.graphics
module 的 StackPane
,但由于发布的异常而失败当您添加Label
时,它属于javafx.controls
module。
这已经说过很多次了:您需要为您的项目设置所需的 VM 选项。
首先阅读https://openjfx.io/openjfx-docs/ 的文档,包括IntelliJ 文档,您的IDE 的非模块化项目部分。并阅读第 4 部分。添加 VM 选项。
所以点击运行 -> 编辑配置,然后添加:
-p /usr/lib/jvm/javafx-sdk-11.0.2/lib --add-modules javafx.controls
应用,运行,问题就解决了。
【讨论】:
有没有办法在不指定虚拟机选项的情况下做同样的事情? @user2914191从 Maven 窗口执行javafx:run
。
不下载 JavaFX SDK 或库,另一种选择是使用包含 JavaFX/OpenJFX 库的 JDK。至少有两个这样的产品:来自 BellSoft 的 LibericaFX,以及来自 Azul Systems 的 Azul Platform Core 构建的 JDK FX。【参考方案2】:
我会为此使用 gradle 和 javafx 插件。
apply plugin: "org.openjfx.javafxplugin"
javafx
version = "11"
modules = ["javafx.base", "javafx.controls", "javafx.graphics"
【讨论】:
【参考方案3】:另一个选项可能有用,尤其是当您计划将应用程序打包为 fat-jar 时。
感谢this answer。
您可以创建另一个不扩展 javafx.application.Application
的 Main
类,而不是修改 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 运行命令(默认情况下会折叠它们)。
【讨论】:
【参考方案4】:检查您的 JAVA_HOME, JAVA_HOME'版本必须是jdk9+或者jdk11+
【讨论】:
以上是关于JavaFX 11:创建标签时出现 IllegalAccessError的主要内容,如果未能解决你的问题,请参考以下文章
idea创建项目时出现Artifact contains illegal characters的解决方法
idea创建项目时出现Artifact contains illegal characters的解决方法
Gradle OpenJFx11:错误 - 缺少 JavaFx 运行时组件