启动 JavaFX 应用程序时的内部 NPE

Posted

技术标签:

【中文标题】启动 JavaFX 应用程序时的内部 NPE【英文标题】:Internal NPE when launching JavaFX Application 【发布时间】:2019-12-27 11:58:44 【问题描述】:

所以基本上我开始了一个虚拟的 JavaFX 项目,只是为了为我的实际问题实现一个简约的例子。但是现在我什至不能再运行那个简约的项目了,也没有收到足够的错误信息来实际用谷歌搜索它。所以现在,当我运行代码时,我收到了给定的错误堆栈,它不会引导我到任何地方。

我正在使用 IntelliJ。 JavaFX 库设置正确,VM 选项设置为: --module-path "C:\Program Files\Java\javafx-sdk-11.0.2\lib" --add-modules javafx.controls,javafx.fxml

最重要的是,当我运行代码时,控制台中会弹出这些错误,但应用程序似乎仍在运行,因为我需要按下 IntelliJ 的红色停止按钮才能真正停止它。

有没有人猜测,这里出了什么问题?我没有足够的经验来跟踪这些错误,因为它们并不指向我的代码,而是指向一些 Deep 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;
    

左.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");
    

【问题讨论】:

看不到任何明显错误的东西(你的主控制器是什么?)——这使得正常的调试路径可以追踪未初始化字段的确切位置;)如果它是一个模块化项目,请确保从您的项目模块中导出(或至少打开)所有相关包 @kleopatra 我添加了 MainWindowController 代码。我如何能够跟踪该错误代码?当我从错误中单击这些代码引用时,我立即陷入了一些深度嵌套的文件代码片段中,例如 Method.java 或类似文件。 如何调用它? main 方法应该是静态的。否则,如果 xml 文件和 java 文件共享同一个文件夹,它应该可以正常运行。 【参考方案1】:

解决方案

您遇到的错误是由您的 main(String[]) 方法不是 static 引起的。如果你把它设为static,那么错误就会消失。


一些解释

只要主类是Application 的子类,JavaFX 就可以在不提供主方法的情况下启动应用程序。但是,开发人员仍然可以包含一个 main 方法,这意味着这个 special launch functionality 必须优雅地处理这种情况。换句话说,从开发人员的角度来看,Application 子类中的显式 main 方法必须充当应用程序的入口点。尽管如此,在幕后,一些深层的内部类已经成为“真正的”主类。

为此,main 方法(如果存在的话)通过Class#getMethod(String,Class...) 定位,虽然它只返回公共方法,但不区分静态和非静态方法。如果找到,Method#invoke(Object,Object...) 用于反射调用 main 方法。 invoke 的第一个参数是应该调用该方法的实例;对于静态方法,值为null。不幸的是,代码假定它找到的方法是静态的,这会导致NullPointerException被抛出——你不能在null“实例”上调用实例方法。


更新:此问题已在 GitHub (#570) 和随后的 JBS (JDK-8230119) 上提交。当前的想法是发出警告而不是抛出NullPointerException。但是,允许在没有 main 方法的情况下启动的功能可能会在未来的版本中被弃用,这将影响该问题的解决方式。

【讨论】:

@r.roger 没问题。也对错误感到困惑,哈哈。不得不深入研究代码,看看为什么会抛出这样一个无用的异常。如果您有兴趣,我添加了有关该问题的一些解释。 @Slaw 也谢谢你的解释。尽管我对更好地理解所有 Java(FX) 内容非常感兴趣,但我并不真正理解这种解释。 :D 我想只是需要更多的时间、精力和经验才能变得如此深入:) 我会坚持下去。

以上是关于启动 JavaFX 应用程序时的内部 NPE的主要内容,如果未能解决你的问题,请参考以下文章

计算 apk 大小时的 NPE

隐藏阶段时的JavaFX 8内存泄漏

android.content.Context.getString 中的 NPE 导致应用程序在启动时崩溃

错误: 在类中找不到 main 方法, 请将 main 方法定义为:    public static void main(String[] args) 否则 JavaFX 应用程

错误: 在类中找不到 main 方法, 请将 main 方法定义为:    public static void main(String[] args) 否则 JavaFX 应用程

使用 glfwWindowShouldClose 时的 NPE (Kotlin)