JavaFX 11: IllegalAccessError при создании метки

Вероятно, этот вопрос связан с той же проблемой, что и этот, но похоже, что спрашивающий не добавил достаточно информации, чтобы получить полезный ответ.

Я пытаюсь запустить приложение JavaFx с JDK и JavaFx SDK версии 11.0.2.

Этот код работает точно так, как ожидалось, создавая пустое окно:

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);
    }
}

Трассировка стека, которую он производит, выглядит следующим образом (строка 13 в Main - это место, где создается метка):

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

Я использую последнюю версию IntelliJ Idea на Kubuntu. Я скачал 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?


person Reinis Mazeiks    schedule 21.01.2019    source источник
comment
При работе с модулями нельзя использовать объявление classpath, и наоборот.   -  person Dragas    schedule 13.12.2020


Ответы (4)


Вы уже даете объяснение своей проблемы:

Я не изменял параметры виртуальной машины.

Поскольку JavaFX 11 больше не является частью JDK, вы должны использовать JavaFX SDK (как вы это делаете) из здесь или, в качестве альтернативы, используйте Maven / Gradle для получения модулей JavaFX из Maven Central.

Затем вам нужно добавить SDK в качестве библиотеки, чтобы IntelliJ мог найти классы JavaFX.

Но как только вы это сделаете, и учитывая, что jar-файлы JavaFX являются модулями, вам все равно нужно сделать две вещи:

  • Сделайте модули JavaFX доступными для вашего module-path
  • Определите, какие модули вы добавляете в проект

Основываясь на вашем выводе 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)

Это объясняет, почему ваш проект запускается в первом случае, когда у вас нет элемента управления, добавленного в сцену, только StackPane, который принадлежит javafx.graphics module, но при добавлении Label, принадлежащего javafx.controls модуль.

Об этом неоднократно говорилось: вам нужно установить необходимые параметры виртуальной машины для вашего проекта.

Начните с прочтения документации на странице https://openjfx.io/openjfx-docs/, включая IntelliJ doc, раздел Немодульные проекты для вашей IDE. И прочтите часть 4. Добавьте параметры ВМ.

Параметры ВМ

Итак, нажмите Run -> Edit Configurations и добавьте:

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

Подайте заявку и запустите, проблема будет решена.

person José Pereda    schedule 21.01.2019
comment
есть ли способ сделать то же самое без указания параметров виртуальной машины? - person user2914191; 15.02.2019
comment
@ user2914191 - Выполните javafx:run из окна Maven. - person ItachiUchiha; 02.08.2019
comment
Вместо загрузки JavaFX SDK или библиотек можно использовать другой вариант - JDK, который включает библиотеки JavaFX / OpenJFX. По крайней мере, два таких продукта: LibericaFX от BellSoft и JDK FX сборки Azul Platform Core от Azul Systems. - person Basil Bourque; 05.06.2021

Я бы использовал для этого gradle и плагин javafx.

apply plugin: "org.openjfx.javafxplugin"

javafx {
    version = "11"
    modules = ["javafx.base", "javafx.controls", "javafx.graphics"
}
person Beowolve    schedule 13.10.2020

Другой вариант может быть полезен, особенно если вы планируете упаковать свое приложение как fat-jar.

Благодаря этому ответу.

Вместо изменения параметров vm вы можете создать другой Main класс, который не расширяет javafx.application.Application, и вызывать оттуда метод запуска вашего приложения. Попробуйте следующее:

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 (по умолчанию идея складывает их), найденные в окнах «Выполнить» или «Отладка».

person Boris    schedule 30.03.2021

проверьте, что ваша версия JAVA_HOME, JAVA_HOME должна быть jdk9 + или jdk11 +

person 远星河    schedule 25.05.2020