我正在尝试使用maven依赖创建一个JFX11自包含jar。从我所做的研究来看,最好的方法似乎是通过maven shade插件。然而,当我运行它时,我得到了这个错误:
错误:缺少运行此应用程序所需的JavaFX运行时组件
我不明白为什么会发生这种事。我搞砸了什么?有没有更好的方法来做这件事?我也用同样的消息尝试了maven汇编插件。
可供参考的pom文件
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>Application</groupId>
<artifactId>Main</artifactId>
<packaging>jar</packaging>
<version>1.0-SNAPSHOT</version>
<name>SpaceRunner</name>
<url>http://maven.apache.org</url>
<dependencies>
<dependency>
<groupId>org.openjfx</groupId>
<artifactId>javafx-controls</artifactId>
<version>11</version>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<version>3.8.0</version>
<configuration>
<release>10</release>
</configuration>
</plugin>
<plugin>
<groupId>org.codehaus.mojo</groupId>
<artifactId>exec-maven-plugin</artifactId>
<version>1.6.0</version>
<executions>
<execution>
<goals>
<goal>java</goal>
</goals>
</execution>
</executions>
<configuration>
<mainClass>Application.Main</mainClass>
</configuration>
</plugin>
<plugin>
<artifactId>maven-jar-plugin</artifactId>
<configuration>
<archive>
<manifest>
<mainClass>
Application.Main
</mainClass>
</manifest>
</archive>
</configuration>
</plugin>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-shade-plugin</artifactId>
<version>3.2.0</version>
<executions>
<execution>
<phase>package</phase>
<goals>
<goal>shade</goal>
</goals>
<configuration>
<transformers>
<transformer implementation="org.apache.maven.plugins.shade.resource.ManifestResourceTransformer">
<mainClass>Application.Main</mainClass>
</transformer>
</transformers>
</configuration>
</execution>
</executions>
</plugin>
</plugins>
</build>
</project>
发布于 2018-10-05 04:37:17
更新10/2021
从JavaFX 16开始,当JavaFX不在模块路径上运行时会显示警告,这是超级/fat jar的情况:
$ java -jar myFatJar-1.0-SNAPSHOT.jar
Oct 02, 2021 1:45:21 PM com.sun.javafx.application.PlatformImpl startup
WARNING: Unsupported JavaFX configuration: classes were loaded from 'unnamed module @14c24f4c'
另外,你会收到一个来自shade插件本身的警告:
[WARNING] Discovered module-info.class. Shading will break its strong encapsulation.
虽然这些警告最初可以忽略,但这是有原因的。
正如此CSR中所解释的
和:
当JavaFX类从类路径加载时,它破坏了封装,因为我们不再从java模块系统中获益。
原始答案
此answer解释了为什么JavaFX 11上的胖/超jar失败。简而言之:
此错误来自java.base模块中的sun.launcher.LauncherHelper。这是因为Main应用程序扩展了Application,并且有一个main方法。如果是这样,LauncherHelper将检查
javafx.graphics
模块是否以命名模块的形式存在。如果该模块不存在,启动将中止。
并且已经为Gradle提出了一个修复方案。
对于Maven,解决方案是完全相同的:提供一个不是从Application
扩展的新主类。
你的application
包中会有一个新的类(坏名字):
// NewMain.java
public class NewMain {
public static void main(String[] args) {
Main.main(args);
}
}
和您现有的Main
类,如下所示:
//Main.java
public class Main extends Application {
@Override
public void start(Stage stage) {
...
}
public static void main(String[] args) {
launch(args);
}
}
现在你需要修改你的pom并为不同的插件设置你的主类:
<mainClass>application.NewMain</mainClass>
特定于平台的Fat jar
最后,使用shade插件,您将在您的机器上生成一个胖jar。
这意味着,到目前为止,您的JavaFX依赖项正在使用唯一的分类器。例如,如果你在Windows上,Maven将在内部使用win
分类器。这样做的效果是只包含Windows的本机库。
所以您使用的是:
的本机dll
现在,如果您生成了fat jar,您将捆绑所有这些依赖项(以及来自您的项目的其他常规第三方依赖项),并且您将能够以如下方式运行您的项目:
java -jar myFatJar-1.0-SNAPSHOT.jar
虽然这很好,但如果您想分发您的jar,请注意这个jar不是跨平台的,它只能在您的平台上工作,在本例中为Windows。
跨平台Fat Jar
有一种解决方案可以生成您可以分发的跨平台jar :包含其他平台的其余本地库。
这很容易做到,因为您只需要包括这三个平台的图形模块依赖项:
<dependencies>
<dependency>
<groupId>org.openjfx</groupId>
<artifactId>javafx-controls</artifactId>
<version>11</version>
</dependency>
<dependency>
<groupId>org.openjfx</groupId>
<artifactId>javafx-graphics </artifactId>
<version>11</version>
<classifier>win</classifier>
</dependency>
<dependency>
<groupId>org.openjfx</groupId>
<artifactId>javafx-graphics </artifactId>
<version>11</version>
<classifier>linux</classifier>
</dependency>
<dependency>
<groupId>org.openjfx</groupId>
<artifactId>javafx-graphics </artifactId>
<version>11</version>
<classifier>mac</classifier>
</dependency>
</dependencies>
大小
这种方法有一个主要问题:大小。正如您在另一个answer中看到的,如果您使用WebView控件,由于WebKit本机库的缘故,您将占用大约220MB。
https://stackoverflow.com/questions/52653836
复制相似问题