java exec source报错_在 Intellij IDEA 里使用 OpenJFX (JavaFX)

本文介绍了如何在IntelliJ IDEA 2018.2上配置和使用OpenJFX,特别是在JDK 11中。通过添加OpenJFX依赖到maven项目,并配置模块化Java程序,解决启动和编译时可能出现的问题。常见错误包括缺少JavaFX运行时组件和-source 8不支持模块。解决方案包括使用非模块化Java程序启动方式、修改pom.xml或创建独立启动类。
摘要由CSDN通过智能技术生成

JDK 11 把 JavaFX 剥离了出来,形成了单独且开源的 OpenJFX 模块。

本文的目的是通过简单的例子解释这一变化对使用 JavaFX 所造成的影响,并找到一种在 IDEA 2018.2 上使用它的办法。

首先,OpenJFX 官网的入门文档指示我们手动下载 SDK,但在 maven 的帮助下这不是必须的。虽然同样得下载,但这被 maven 自动化了。

我们的 pom.xml 如下:

xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">

4.0.0

sample

javafx

1.0-SNAPSHOT

UTF-8

sample.JFXMain

11

org.openjfx

javafx-controls

${javafx.version}

org.openjfx

javafx-fxml

${javafx.version}

org.apache.maven.plugins

maven-compiler-plugin

3.8.0

11

11

这里引入了 OpenJFX 的依赖包,并设置了项目的 JDK 版本为 JDK 11。

根据 IDEA 的提示 Import Changes,或者手动:右键 pom.xml - Maven - Reimport。

注意:这里没有使用 maven.compiler.source 和 maven.compiler.target 这两个 property。这两个 property 是作为参数定义在 maven-compiler-plugin 里的,分别对应于源代码的 Java 版本和目标代码的 Java 版本。因为 IDEA 目前对这两项的支持似乎不够好,不能完美地同步到项目设置里。

注意:需要手动检查 Preferences - Build, Execution, Deployment - Compiler - Java Compiler 中 Project bytecode version 及 Per-module bytecode version 的值是否同为 11。理由同上。

我们的 Java 代码如下:

package sample;

import javafx.application.Application;

import javafx.fxml.FXMLLoader;

import javafx.scene.Parent;

import javafx.scene.Scene;

import javafx.stage.Stage;

public class JFXMain extends Application {

@Override

public void start(Stage primaryStage) throws Exception{

Parent root = FXMLLoader.load(getClass().getResource("/sample.fxml"));

primaryStage.setTitle("Hello World");

primaryStage.setScene(new Scene(root, 300, 275));

primaryStage.show();

}

public static void main(String[] args) {

launch(args);

}

}

package sample;

public class Controller {

}

OpenJFX 布局描述文件 /src/main/resources/sample.fxml 如下:

xmlns:fx="http://javafx.com/fxml" alignment="center" hgap="10" vgap="10">

Java 模块描述文件 /src/main/java/module-info.java 如下:

module sample {

requires javafx.controls;

requires javafx.fxml;

// 暴露包 sample 给 javafx 的模块们,使其可以在运行时使用反射访问

opens sample to javafx.graphics, javafx.fxml;

}

以上便是在 JDK 11 中使用 OpenJFX 所需的全部铺垫了。

常见错误

启动报错:缺少 JavaFX 运行时组件, 需要使用该组件来运行此应用程序

模块化 Java 程序与非模块化 Java 程序的启动方式有所不同。

# 非模块化

java [options] mainclass [args...]

# 模块化

java [options] [--module-path modulepath] --module module[/mainclass] [args...]

提供了 module-info.java 的话,IDEA 发现这是模块化的 Java 程序。以上例为例,启动命令是:

java ${OPTIONS} -m ${METHOD_PATH} -m sample/sample.JFXMain

否则,IDEA 会认为这是非模块化 Java 程序,启动命令是:

java ${OPTIONS} -classpath ${CLASS_PATH} sample.JFXMain

但这报错具体是什么代码引起的呢?我们在 JDK 11 的 sun.launcher.LauncherHelper 发现:如果 JFXMain 继承自 javafx.application.Application,同时程序从 JFXMain.main() 启动,LauncherHelper 会检查是否存在模块 javafx.graphics 的声明:

package sun.launcher;

public final class LauncherHelper {

static final class FXHelper {

private static void setFXLaunchParameters(String what, int mode) {

...

Optional om = ModuleLayer.boot().findModule(JAVAFX_GRAPHICS_MODULE_NAME);

if (!om.isPresent()) {

abort(null, "java.launcher.cls.error5");

}

...

}

}

}

显然,如果不以模块化 Java 程序的方式启动,没有模块信息。错误码 java.launcher.cls.error5 即为 “错误: 缺少 JavaFX 运行时组件, 需要使用该组件来运行此应用程序。”

不过我们还有其他办法来绕开 LauncherHelper 的检查,能够以非模块化 Java 程序的方式运行程序。思路是:使程序的入口 main() 不继承自 javafx.application.Application。

因此,我们可以使用 maven 来运行程序,因为 maven 的 main() 显然满足该要求。这用到了 exec-maven-plugin,这个插件是默认包含的,我们可以直接使用它的 property exec.mainClass。

修改 pom.xml:

...

sample.JFXMain

...

运行命令如下:

mvn clean compile exec:java

除此之外,我们也可以单独创建一个启动类:

package sample;

import javafx.application.Application;

public class AppMain {

public static void main(String[] args) {

Application.launch(JFXMain.class, args);

}

}

从这个类启动 Java 程序,效果相同。

编译报错:Error: (4, 1) java: -source 8 中不支持 模块

根据上文所述的方法,检查并修改 Project bytecode version。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值