[英]JavaFX 11: IllegalAccessError when creating Label
這個問題可能是同一個問題的這一個,但現在看來,一個的提問者還沒有添加足夠的信息來獲得一個有用的響應。
我正在嘗試使用 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 行是創建 Label 的位置):
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 中的錯誤嗎?
您已經在解釋您的問題:
我沒有修改任何 VM 選項。
由於 JavaFX 11 不再是 JDK 的一部分,因此您必須從這里使用 JavaFX SDK(正如您所做的那樣),或者使用 Maven/Gradle 從 Maven Central 檢索 JavaFX 模塊。
然后需要將 SDK 添加為庫,以便 IntelliJ 可以找到 JavaFX 類。
但是一旦你這樣做了,並且考慮到 JavaFX jars 是模塊,你仍然需要做兩件事:
根據您的 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
模塊的StackPane
,但是當您添加Label
時會因發布的異常而失敗,即屬於javafx.controls
模塊。
這已經說過很多次了:您需要為您的項目設置所需的 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
應用,運行,問題就解決了。
我會為此使用 gradle 和 javafx 插件。
apply plugin: "org.openjfx.javafxplugin"
javafx {
version = "11"
modules = ["javafx.base", "javafx.controls", "javafx.graphics"
}
另一個選項可能很有用,尤其是當您計划將應用程序打包為 fat-jar 時。
感謝這個答案。
您可以創建另一個不擴展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);
}
}
嘗試使用默認配置( VM options
為空字段)運行App.main()
會導致錯誤,而Runner.main()
執行成功。
為了更深入地理解,您可以比較在 Run 或 Debug 窗口中找到的 java 運行命令(idea 默認折疊它們)。
檢查你的JAVA_HOME,JAVA_HOME'版本必須是jdk9+或jdk11+
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.