jdeps не работает с Java 11 и JavaFX в толстой банке

Я пытаюсь использовать jdeps (и jlink) для развертывания моего приложения Java 11, которое использует JavaFX 11. Все работает нормально внутри моей Java IDE. Но когда я использую jdep, я получаю следующую ошибку, которая указывает на то, что не удается найти какой-либо модуль или пакет. Я застрял. Спасибо за вашу помощь.

jdeps --list-deps --module-path /Users/…/target:/Library/Java/JavaVirtualMachines/jdk-11.0.2.jdk/Contents/Home/jmods:/Users/…/javafx-sdk-11.0. 2 / lib --add-modules chaincoder, javafx.fxml, javafx.base, javafx.controls, javafx.graphics, javafx.web /Users/…/target/chaincoder4-1.0.jar

Error: Module javafx.media contains package javafx.collections, module javafx.base exports package javafx.collections to javafx.media

module-info.java - это

module chaincoder {

requires   javafx.web;
requires   javafx.graphics;
requires   javafx.controls;
requires   javafx.fxml;
requires   javafx.base;
requires   javafx.media;

requires java.desktop;
requires java.base;
requires java.xml;
requires java.logging;

requires jdk.jsobject;

exports   core;

}

pom.xml - это

        <dependency>
            <groupId>org.openjfx</groupId>
            <artifactId>javafx-graphics</artifactId>
            <version>11.0.2</version>
        </dependency>

        <dependency>
            <groupId>org.openjfx</groupId>
            <artifactId>javafx-controls</artifactId>
            <version>11.0.2</version>
        </dependency>

        <dependency>
            <groupId>org.openjfx</groupId>
            <artifactId>javafx-base</artifactId>
            <version>11.0.2</version>
        </dependency>

        <dependency>
            <groupId>org.openjfx</groupId>
            <artifactId>javafx-media</artifactId>
            <version>11.0.2</version>
        </dependency>

        <dependency>
            <groupId>org.openjfx</groupId>
            <artifactId>javafx-web</artifactId>
            <version>11.0.2</version>
        </dependency>

        <dependency>
            <groupId>org.openjfx</groupId>
            <artifactId>javafx-fxml</artifactId>
            <version>11.0.2</version>
        </dependency>


<build>
    <plugins>

        <plugin>
            <!-- Build an executable JAR -->
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-jar-plugin</artifactId>
            <version>3.1.0</version>
            <configuration>
                <archive>
                    <manifest>
                        <addClasspath>true</addClasspath>
                        <classpathPrefix>lib/</classpathPrefix>
                        <mainClass>core.Main</mainClass>
                    </manifest>
                </archive>
            </configuration>
        </plugin>

        <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-dependency-plugin</artifactId>
            <version>2.6</version>
            <executions>
                <execution>
                    <id>unpack-dependencies</id>
                    <phase>package</phase>
                    <goals>
                        <goal>unpack-dependencies</goal>
                    </goals>
                    <configuration>
                        <excludeScope>system</excludeScope>
                        <excludeGroupIds>junit,org.mockito,org.hamcrest</excludeGroupIds>
                        <outputDirectory>${project.build.directory}/classes</outputDirectory>
                    </configuration>
                </execution>
            </executions>
        </plugin>

        <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-compiler-plugin</artifactId>
            <version>3.8.0</version>
            <configuration>
                <source>11</source>
                <target>11</target>
                <compilerArguments>
                    <bootclasspath>${sun.boot.class.path}${path.separator}${java.home}/lib/jfxrt.jar</bootclasspath>
                </compilerArguments>
            </configuration>
        </plugin> 
    </plugins>

</build>

person cfsoe    schedule 18.02.2019    source источник


Ответы (1)


По какой причине вы используете плагин maven-dependency-plugin?

Если вы увидите, что делает цель unpack-dependencies, вы заметите, что папки target/classes содержат более 90 МБ классов после того, как плагин извлечет все зависимости проекта, включая зависимости JavaFX, и объединит их все.

И угадайте, что происходит с разными module-info.class файлами для каждого из разных модулей JavaFX? Преобладает только один, в данном случае - от javafx.media.

Итак, вы создали новый модуль в target/classes!

Он называется javafx.media, содержит классы вашего проекта и все классы JavaFX, включая все из javafx.base или javafx.graphics.

Если вы проверите этот target/classes/module-info.class файл:

module javafx.media {
    requires transitive javafx.base;
    requires transitive javafx.graphics;
    ...
}

этот «модуль», который уже содержит все классы JavaFX, также потребует снова этих классов из javafx.base и javafx.graphics, дублируя их.

Это объясняет ваше сообщение об ошибке:

Error: Module javafx.media contains package javafx.collections, 
       module javafx.base exports package javafx.collections to javafx.media

Теперь вы можете видеть, что ваш новый модуль javafx.media содержит множество пакетов, которые также экспортируются из javafx.base в него, и это нарушает разделить пакет не разрешено условие:

Один и тот же пакет Java может быть экспортирован только одним модулем Java во время выполнения. Другими словами, у вас не может быть двух (или более) модулей, которые экспортируют один и тот же пакет в одно и то же время. Если вы это сделаете, виртуальная машина Java будет жаловаться при запуске.

Возможные решения

  • Может быть, вам вообще не нужен плагин maven-dependency-plugin? Удалите его, если он вам не нужен.

  • Вы добавляете папку target со всеми классами JavaFX в путь к модулю:

    jdeps --list-deps --module-path /Users/…/target:...
    

Очевидно, это неправильно, поскольку папка classes содержит все классы JavaFX и неправильный дескриптор информации о модуле. Это недопустимый модуль, поэтому просто удалите его из пути к модулю.

  • Тогда вам не нужно добавлять путь JDK к модулю-пути, это делается по умолчанию.

Таким образом, вы можете просто:

jdeps --list-deps --module-path /Users/…/javafx-sdk-11.0.2/lib
  • Затем, когда вы добавляете модули, вам не нужно добавлять javafx.base или javafx.graphics, поскольку они включены из транзитивных зависимостей (это также относится к зависимостям, перечисленным в вашем pom). См., Например, javafx.web.

Итак, у вас будет:

jdeps --list-deps --module-path /Users/…/javafx-sdk-11.0.2/lib
    -add-modules chaincoder,javafx.fxml,javafx.web
  • Но обратите внимание, что у вас есть модуль с именем chaincoder, который уже включает в себя все необходимые модули, поэтому вы можете иметь только это:

    jdeps --list-deps --module-path /Users/…/javafx-sdk-11.0.2/lib
        -add-modules chaincoder /Users/…/target/chaincoder4-1.0.jar
    

Это должно сработать.

В заключение, вы можете избавиться от jfxrt.jar из плагина компилятора:

<configuration>
    <source>11</source>
    <target>11</target>
    <compilerArguments>
            <bootclasspath>${sun.boot.class.path}${path.separator}${java.home}/lib/jfxrt.jar</bootclasspath>
    </compilerArguments>
</configuration>

Поскольку его больше нет. У вас есть все классы JavaFX, включенные в зависимости:

<configuration>
    <source>11</source>
    <target>11</target>
</configuration>
person José Pereda    schedule 18.02.2019