启动 JavaFX 应用程序时的内部 NPE
Internal NPE when launching JavaFX Application
所以基本上我开始了一个虚拟 JavaFX 项目只是为了为我的实际问题实现一个简约的例子。但现在我什至无法再 运行 那个简约的项目,也没有收到足够的错误信息来实际 google 自己解决这个问题。所以现在,当我 运行 代码时,我收到给定的错误堆栈,它不会引导我到任何地方。
我正在使用 IntelliJ。 JavaFX 库设置正确,VM 选项设置为:
--module-path "C:\Program Files\Java\javafx-sdk-11.0.2\lib" --add-modules javafx.controls,javafx.fxml
最重要的是,当我 运行 代码时,这些错误会在控制台中弹出,但应用程序似乎仍处于 运行ning 状态,因为我需要按下 IntelliJ 的红色停止按钮真正阻止它。
有没有人猜到这里出了什么问题?我没有足够的经验来跟踪这些错误,因为它们没有指向我的代码,而是指向一些深层 Java 代码。
异常:
Exception in thread "main" 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 java.base/sun.launcher.LauncherHelper$FXHelper.main(LauncherHelper.java:1051)
Caused by: java.lang.NullPointerException
at java.base/java.lang.reflect.Method.invoke(Method.java:559)
at javafx.graphics/com.sun.javafx.application.LauncherImpl.launchApplicationWithArgs(LauncherImpl.java:464)
at javafx.graphics/com.sun.javafx.application.LauncherImpl.launchApplication(LauncherImpl.java:363)
... 5 more
Main.java:
package sample;
import javafx.application.Application;
import javafx.fxml.FXMLLoader;
import javafx.scene.Scene;
import javafx.scene.layout.AnchorPane;
import javafx.scene.layout.BorderPane;
import javafx.stage.Stage;
import java.io.IOException;
public class Main extends Application {
private Stage rootStage;
public BorderPane mainWindow;
public AnchorPane left;
public AnchorPane bottom;
@Override
public void start(Stage primaryStage) throws Exception {
this.rootStage = primaryStage;
loadMainWindow();
}
public void loadMainWindow() throws IOException {
FXMLLoader loaderMainWindow = new FXMLLoader(Main.class.getResource("MainWindow.fxml"));
mainWindow = loaderMainWindow.load();
FXMLLoader loaderLeft = new FXMLLoader(Main.class.getResource("Left.fxml"));
left = loaderLeft.load();
mainWindow.setLeft(left);
//mainWindow.setBottom(bottom);
Scene scene = new Scene(mainWindow);
rootStage.setScene(scene);
rootStage.show();
}
public void main(String[] args) {
launch(args);
}
}
MainWindow.fxml:
<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.scene.layout.BorderPane?>
<BorderPane prefHeight="400.0" prefWidth="600.0" xmlns="http://javafx.com/javafx/10.0.1" xmlns:fx="http://javafx.com/fxml/1" fx:controller="sample.MainWindowController" />
主窗口控制器:
package sample;
import javafx.fxml.Initializable;
import java.net.URL;
import java.util.ResourceBundle;
public class MainWindowController implements Initializable {
private Main main;
@Override
public void initialize(URL url, ResourceBundle resourceBundle) {
}
public void setMain(Main main) {
this.main = main;
}
}
Left.fxml:
<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.scene.control.Button?>
<?import javafx.scene.layout.AnchorPane?>
<AnchorPane prefHeight="400.0" prefWidth="100.0" xmlns="http://javafx.com/javafx/10.0.1" xmlns:fx="http://javafx.com/fxml/1" fx:controller="sample.LeftController">
<children>
<Button fx:id="button" layoutX="237.0" layoutY="169.0" mnemonicParsing="false" onAction="#buttonClick" text="Button" AnchorPane.leftAnchor="0.0" AnchorPane.rightAnchor="0.0" />
</children>
</AnchorPane>
LeftController.java:
package sample;
import javafx.fxml.FXML;
import javafx.fxml.Initializable;
import javafx.scene.control.Button;
import java.awt.event.ActionEvent;
import java.net.URL;
import java.util.ResourceBundle;
public class LeftController implements Initializable {
@FXML
private Button button;
@Override
public void initialize(URL url, ResourceBundle resourceBundle) {
}
public void buttonClick(javafx.event.ActionEvent actionEvent) {
System.out.println("Some Stuff");
}
}
解决方案
您遇到的错误是因为您的 main(String[])
方法不是 static
。如果你做到了 static
那么错误就会消失。
一些解释
JavaFX 提供了无需提供 main 方法即可启动应用程序的能力,只要主要 class 是 Application
的子 class。但是,开发人员仍然可以包含一个 main 方法,这意味着 必须优雅地处理这种情况。换句话说,Application
subclass 中存在的显式 main 方法必须从开发人员的角度充当应用程序的入口点。尽管如此,在幕后一些深层的内部 class 已经成为 "real" 主要 class.
为此,main 方法位于(如果存在的话)via Class#getMethod(String,Class...)
which, while only returning public methods, doesn't distinguish between static and non-static methods. If found, Method#invoke(Object,Object...)
用于反射调用 main 方法。 invoke
的第一个参数是应该调用该方法的实例;在静态方法的情况下,该值为 null
。不幸的是,代码 假定 它找到的方法是静态的,这会导致抛出 NullPointerException
— 你不能在 null
"instance".
更新: 此问题已在 GitHub (#570) and subsequently JBS (JDK-8230119) 上提交。当前的想法是发出警告而不是抛出 NullPointerException
。但是,允许在没有 main 方法的情况下启动的功能可能会在未来的版本中被弃用,这将影响此问题的解决方式。
所以基本上我开始了一个虚拟 JavaFX 项目只是为了为我的实际问题实现一个简约的例子。但现在我什至无法再 运行 那个简约的项目,也没有收到足够的错误信息来实际 google 自己解决这个问题。所以现在,当我 运行 代码时,我收到给定的错误堆栈,它不会引导我到任何地方。
我正在使用 IntelliJ。 JavaFX 库设置正确,VM 选项设置为:
--module-path "C:\Program Files\Java\javafx-sdk-11.0.2\lib" --add-modules javafx.controls,javafx.fxml
最重要的是,当我 运行 代码时,这些错误会在控制台中弹出,但应用程序似乎仍处于 运行ning 状态,因为我需要按下 IntelliJ 的红色停止按钮真正阻止它。
有没有人猜到这里出了什么问题?我没有足够的经验来跟踪这些错误,因为它们没有指向我的代码,而是指向一些深层 Java 代码。
异常:
Exception in thread "main" 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 java.base/sun.launcher.LauncherHelper$FXHelper.main(LauncherHelper.java:1051)
Caused by: java.lang.NullPointerException
at java.base/java.lang.reflect.Method.invoke(Method.java:559)
at javafx.graphics/com.sun.javafx.application.LauncherImpl.launchApplicationWithArgs(LauncherImpl.java:464)
at javafx.graphics/com.sun.javafx.application.LauncherImpl.launchApplication(LauncherImpl.java:363)
... 5 more
Main.java:
package sample;
import javafx.application.Application;
import javafx.fxml.FXMLLoader;
import javafx.scene.Scene;
import javafx.scene.layout.AnchorPane;
import javafx.scene.layout.BorderPane;
import javafx.stage.Stage;
import java.io.IOException;
public class Main extends Application {
private Stage rootStage;
public BorderPane mainWindow;
public AnchorPane left;
public AnchorPane bottom;
@Override
public void start(Stage primaryStage) throws Exception {
this.rootStage = primaryStage;
loadMainWindow();
}
public void loadMainWindow() throws IOException {
FXMLLoader loaderMainWindow = new FXMLLoader(Main.class.getResource("MainWindow.fxml"));
mainWindow = loaderMainWindow.load();
FXMLLoader loaderLeft = new FXMLLoader(Main.class.getResource("Left.fxml"));
left = loaderLeft.load();
mainWindow.setLeft(left);
//mainWindow.setBottom(bottom);
Scene scene = new Scene(mainWindow);
rootStage.setScene(scene);
rootStage.show();
}
public void main(String[] args) {
launch(args);
}
}
MainWindow.fxml:
<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.scene.layout.BorderPane?>
<BorderPane prefHeight="400.0" prefWidth="600.0" xmlns="http://javafx.com/javafx/10.0.1" xmlns:fx="http://javafx.com/fxml/1" fx:controller="sample.MainWindowController" />
主窗口控制器:
package sample;
import javafx.fxml.Initializable;
import java.net.URL;
import java.util.ResourceBundle;
public class MainWindowController implements Initializable {
private Main main;
@Override
public void initialize(URL url, ResourceBundle resourceBundle) {
}
public void setMain(Main main) {
this.main = main;
}
}
Left.fxml:
<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.scene.control.Button?>
<?import javafx.scene.layout.AnchorPane?>
<AnchorPane prefHeight="400.0" prefWidth="100.0" xmlns="http://javafx.com/javafx/10.0.1" xmlns:fx="http://javafx.com/fxml/1" fx:controller="sample.LeftController">
<children>
<Button fx:id="button" layoutX="237.0" layoutY="169.0" mnemonicParsing="false" onAction="#buttonClick" text="Button" AnchorPane.leftAnchor="0.0" AnchorPane.rightAnchor="0.0" />
</children>
</AnchorPane>
LeftController.java:
package sample;
import javafx.fxml.FXML;
import javafx.fxml.Initializable;
import javafx.scene.control.Button;
import java.awt.event.ActionEvent;
import java.net.URL;
import java.util.ResourceBundle;
public class LeftController implements Initializable {
@FXML
private Button button;
@Override
public void initialize(URL url, ResourceBundle resourceBundle) {
}
public void buttonClick(javafx.event.ActionEvent actionEvent) {
System.out.println("Some Stuff");
}
}
解决方案
您遇到的错误是因为您的 main(String[])
方法不是 static
。如果你做到了 static
那么错误就会消失。
一些解释
JavaFX 提供了无需提供 main 方法即可启动应用程序的能力,只要主要 class 是 Application
的子 class。但是,开发人员仍然可以包含一个 main 方法,这意味着 Application
subclass 中存在的显式 main 方法必须从开发人员的角度充当应用程序的入口点。尽管如此,在幕后一些深层的内部 class 已经成为 "real" 主要 class.
为此,main 方法位于(如果存在的话)via Class#getMethod(String,Class...)
which, while only returning public methods, doesn't distinguish between static and non-static methods. If found, Method#invoke(Object,Object...)
用于反射调用 main 方法。 invoke
的第一个参数是应该调用该方法的实例;在静态方法的情况下,该值为 null
。不幸的是,代码 假定 它找到的方法是静态的,这会导致抛出 NullPointerException
— 你不能在 null
"instance".
更新: 此问题已在 GitHub (#570) and subsequently JBS (JDK-8230119) 上提交。当前的想法是发出警告而不是抛出 NullPointerException
。但是,允许在没有 main 方法的情况下启动的功能可能会在未来的版本中被弃用,这将影响此问题的解决方式。