gpt4 book ai didi

javafx - 如何确定JavaFX应用程序所需的FXML文件,CSS文件,图像和其他资源的正确路径?

转载 作者:行者123 更新时间:2023-12-03 05:25:16 24 4
gpt4 key购买 nike

我的JavaFX应用程序需要能够找到FXML文件,以便使用FXMLLoader以及样式表(CSS文件)和图像来加载它们。当我尝试加载这些项目时,经常会出错,或者我尝试加载的项目根本无法在运行时加载。

对于FXML文件,我看到的错误消息包括

Caused by: java.lang.NullPointerException: location is not set

对于图像,堆栈跟踪包括

Caused by: java.lang.IllegalArgumentException: Invalid URL: Invalid URL or resource not found

如何找出这些资源的正确资源路径?

最佳答案

简短版答案:

  • 使用getClass().getResource(...)SomeOtherClass.class.getResource(...)为资源
  • 创建 URL
  • 将绝对路径(带有前导/)或相对路径(没有前导/)传递给getResource(...)方法。路径是包含资源的包,其中.替换为/
  • 不要在资源路径中使用 ..。如果并且当应用程序 bundle 为jar文件时,这将不起作用。如果资源不在该类的同一包或子包中,请使用绝对路径。
  • 对于FXML文件,将URL直接传递到FXMLLoader
  • 对于图像和样式表,请在toExternalForm()上调用URL生成String,以传递给ImageImageView构造函数,或添加到stylesheets列表中。
  • 要进行故障排除,请检查buid文件夹(或jar文件)的内容,而不是源文件夹的内容。

  • 完整答案
    内容
  • 此答案的范围
  • 资源在运行时加载
  • JavaFX使用URL加载资源
  • 资源名称规则
  • 使用getClass().getResource(...)创建资源URL
  • 组织代码和资源
  • Maven(和类似的)标准布局
  • 故障排除

  • 这个答案的范围
    请注意,此答案仅解决作为应用程序一部分并与之 bundle 在一起的加载资源(例如FXML文件,图像和样式表)的问题。因此,例如,在运行应用程序的计算机上加载用户从文件系统中选择的图像,将需要此处未涵盖的不同技术。

    资源在运行时加载
    了解加载资源的第一件事是,它们当然是在运行时加载的。通常,在开发过程中,应用程序是从文件系统运行的;也就是说,运行该文件所需的类文件和资源是文件系统上的单个文件。但是,一旦构建了应用程序,通常就可以从jar文件中执行该应用程序。在这种情况下,诸如FXML文件,样式表和图像之类的资源不再是文件系统上的单个文件,而是jar文件中的条目。因此:

    Code cannot use File, FileInputStream, or file: URLs to load a resource



    JavaFX使用URL加载资源
    JavaFX使用URL加载FXML,图像和CSS样式表。 FXMLLoader明确希望将 java.net.URL对象传递给它(传递给 static FXMLLoader.load(...)方法,传递给 FXMLLoader构造函数或传递给 setLocation()方法)。 ImageScene.getStylesheets().add(...)都期望表示URL的 String。如果URL是在没有方案的情况下传递的,则它们将相对于类路径进行解释。通过调用 URL上的 toExternalForm(),可以以健壮的方式从 URL创建这些字符串。
    为资源创建正确的URL的推荐机制是使用 Class.getResource(...),它在适当的 Class实例上调用。可以通过调用 getClass()(给出当前对象的类)或 ClassName.class获得此类实例。 Class.getResource(...)方法采用代表资源名称的 String

    资源名称规则
  • 资源名称是/分隔的路径名。每个组件代表一个包或子包名称组件。
  • 资源名称区分大小写。
  • 资源名称中的各个组件必须是有效的Java标识符

  • 最后一点具有重要意义:

    . and .. are not valid Java identifiers, so they cannot be used in resource names.


    当应用程序从文件系统运行时,这些命令实际上可能起作用,尽管这确实是 getResource()实现的偶然事件。当应用程序 bundle 为jar文件时,它们将失败。
    同样,如果您在不能区分仅大小写不同的文件名的操作系统上运行,则从文件系统运行时在资源名称中使用错误的大小写可能会起作用,但从jar文件运行时会失败。
    以前置 /开头的资源名称是绝对的:换句话说,它们相对于类路径进行解释。没有前导 /的资源名称将相对于调用 getResource()的类进行解释。
    这方面的一个小变化是使用 getClass().getClassLoader().getResource(...)。提供给 ClassLoader.getResource(...)的路径始终是绝对路径,即它是相对于classpath的路径。

    使用 getClass().getResource()创建资源URL
    要创建资源URL,请使用 someClass.getResource(...)。通常, someClass表示当前对象的类,并使用 getClass()获得。但是,不一定要如此,如下一节所述。
  • 如果资源与当前类位于同一包中,或者位于该类的子包中,请使用资源的相对路径:
     // FXML file in the same package as the current class:
    URL fxmlURL = getClass().getResource("MyFile.fxml");
    Parent root = FXMLLoader.load(fxmlURL);

    // FXML file in a subpackage called `fxml`:
    URL fxmlURL2 = getClass().getResource("fxml/MyFile.fxml");
    Parent root2 = FXMLLoader.load(fxmlURL2);

    // Similarly for images:
    URL imageURL = getClass().getResource("myimages/image.png");
    Image image = new Image(imageURL.toExternalForm());
  • 如果资源位于不是当前类的子包的包中,请使用绝对路径。例如,如果当前类在org.jamesd.examples.view包中,并且我们需要加载在style.css包中的CSS文件org.jamesd.examples.css,则必须使用绝对路径:
     URL cssURL = getClass().getResource("/org/jamesd/examples/css/style.css");
    scene.getStylesheets().add(cssURL.toExternalForm());
    对于此示例,值得再次强调的是,路径"../css/style.css"不包含有效的Java资源名称,如果应用程序 bundle 为jar文件,则将不起作用

  • 组织代码和资源
    我建议将代码和资源组织到由与它们相关联的UI部分确定的程序包中。 Eclipse中的以下源代码布局提供了此组织的示例:
    enter image description here
    使用这种结构,每个资源在同一包中都有一个类,因此很容易为任何资源生成正确的URL:
    FXMLLoader editorLoader = new FXMLLoader(EditorController.class.getResource("Editor.fxml"));
    Parent editor = editorLoader.load();
    FXMLLoader sidebarLoader = new FXMLLoader(SidebarController.class.getResource("Sidebar.fxml"));
    Parent sidebar = sidebarLoader.load();

    ImageView logo = new ImageView();
    logo.setImage(newImage(SidebarController.class.getResource("logo.png").toExternalForm()));

    mainScene.getStylesheets().add(App.class.getResource("style.css").toExternalForm());
    例如,如果您有一个仅包含资源却没有类的包,则下面布局中的 images
    enter image description here
    您甚至可以考虑仅出于查找资源名称的目的而创建“标记接口(interface)”:
    package org.jamesd.examples.sample.images ;
    public interface ImageLocation { }
    现在,您可以轻松找到这些资源:
    Image clubs = new Image(ImageLocation.class.getResource("clubs.png").toExternalForm());
    从类的子包加载资源也相当简单。给定以下布局:
    enter image description here
    我们可以按以下方式在 App类中加载资源:
    package org.jamesd.examples.resourcedemo;

    import java.net.URL;

    import javafx.application.Application;
    import javafx.fxml.FXMLLoader;
    import javafx.scene.Parent;
    import javafx.scene.Scene;
    import javafx.stage.Stage;

    public class App extends Application {

    @Override
    public void start(Stage primaryStage) throws Exception {

    URL fxmlResource = getClass().getResource("fxml/MainView.fxml");


    FXMLLoader loader = new FXMLLoader();
    loader.setLocation(fxmlResource);
    Parent root = loader.load();
    Scene scene = new Scene(root);
    scene.getStylesheets().add(getClass().getResource("style/main-style.css").toExternalForm());
    primaryStage.setScene(scene);
    primaryStage.show();
    }

    public static void main(String[] args) {
    Application.launch(args);
    }

    }
    要加载与要从中加载资源的类不在同一包或子包中的资源,需要使用绝对路径:
        URL fxmlResource = getClass().getResource("/org/jamesd/examples/resourcedemo/fxml/MainView.fxml");

    Maven(和类似的)标准布局
    Maven和其他依赖项管理和构建工具建议使用源文件夹布局,其中资源与Java源文件分开。上一个示例的Maven布局版本如下所示:
    enter image description here
    重要的是要了解如何构建它来组装应用程序:

    源文件夹 *.java中的
  • src/main/java文件被编译为类文件,这些文件将部署到build文件夹或jar文件中。
  • 资源文件夹src/main/resources中的资源被复制到构建文件夹或jar文件。

  • 在此示例中,由于资源位于与定义了源代码的包的子包相对应的文件夹中,因此生成的内部版本(默认情况下,使用Maven在 target/classes中)由单个结构组成。
    请注意, src/main/javasrc/main/resources均被视为构建中相应结构的根,因此,构建中仅包含其内容,而不是文件夹本身。换句话说,运行时没有 resources文件夹。下面的“疑难解答”部分中显示了构建结构。
    请注意,在这种情况下(Eclipse),IDE显示的 src/main/java源文件夹与 src/main/resources文件夹不同。在第一种情况下,它显示软件包,但对于资源文件夹,它显示文件夹。确保知道您是在IDE中创建包(名称为 .分隔的)或文件夹(名称不得包含 .或在Java标识符中无效的任何其他字符)。

    故障排除
    如果遇到意外错误,请首先检查以下内容:
  • 确保您的资源没有使用无效的名称。这包括在资源路径中使用...
  • 确保在预期的位置使用相对路径,在预期的位置使用绝对路径。对于Class.getResource(...),如果路径具有前导/,则该路径为绝对路径,否则为相对路径。对于ClassLoader.getResource(...),路径始终是绝对的。
  • 请记住,绝对路径是相对于类路径定义的。通常,类路径的根是IDE中所有源和资源文件夹的并集。

  • 如果所有这些似乎都正确,但是您仍然看到错误,请检查构建或部署文件夹。此文件夹的确切位置将因IDE和构建工具而异。如果您使用的是Maven,则默认为 target/classes。其他构建工具和IDE将部署到名为 binclassesbuildout的文件夹中。
    通常,您的IDE不会显示build文件夹,因此您可能需要使用系统文件浏览器进行检查。
    上面的Maven示例的组合的源代码和构建结构是
    enter image description here
    如果要生成jar文件,则某些IDE可能会允许您在树形 View 中展开jar文件以检查其内容。您也可以使用 jar tf file.jar从命令行检查内容:
    $ jar -tf resource-demo-0.0.1-SNAPSHOT.jar 
    META-INF/
    META-INF/MANIFEST.MF
    org/
    org/jamesd/
    org/jamesd/examples/
    org/jamesd/examples/resourcedemo/
    org/jamesd/examples/resourcedemo/images/
    org/jamesd/examples/resourcedemo/style/
    org/jamesd/examples/resourcedemo/fxml/
    org/jamesd/examples/resourcedemo/images/so-logo.png
    org/jamesd/examples/resourcedemo/style/main-style.css
    org/jamesd/examples/resourcedemo/Controller.class
    org/jamesd/examples/resourcedemo/fxml/MainView.fxml
    org/jamesd/examples/resourcedemo/App.class
    module-info.class
    META-INF/maven/
    META-INF/maven/org.jamesd.examples/
    META-INF/maven/org.jamesd.examples/resource-demo/
    META-INF/maven/org.jamesd.examples/resource-demo/pom.xml
    META-INF/maven/org.jamesd.examples/resource-demo/pom.properties
    $
    如果资源没有被部署或正被部署到意外的位置,请检查构建工具或IDE的配置。

    关于javafx - 如何确定JavaFX应用程序所需的FXML文件,CSS文件,图像和其他资源的正确路径?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/61582370/

    24 4 0
    Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
    广告合作:1813099741@qq.com 6ren.com