簡體   English   中英

JavaFX 11:創建標簽時出現 IllegalAccessError

[英]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 是模塊,你仍然需要做兩件事:

  • 使 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相同)

這解釋了為什么您的項目在第一種情況下運行,當您沒有將控件添加到場景中時,只有屬於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.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM