- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我的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(...)
方法。路径是包含资源的包,其中.
替换为/
。
不要在资源路径中使用..
。如果并且当应用程序捆绑为jar文件时,这将无法工作。如果资源不在类的同一包或子包中,请使用绝对路径。
对于FXML文件,将URL
直接传递到FXMLLoader
。
对于图像和样式表,请在toExternalForm()
上调用URL
生成String
,以传递给Image
或ImageView
构造函数,或添加到stylesheets
列表中。
完整答案
内容
这个答案的范围
资源在运行时加载
JavaFX使用URL加载资源
资源名称规则
使用getClass().getResource(...)
创建资源URL
组织代码和资源
Maven(和类似的)标准布局
故障排除
这个答案的范围
请注意,此答案仅解决了作为应用程序一部分并与之捆绑在一起的加载资源(例如FXML文件,图像和样式表)的问题。因此,例如,在运行应用程序的计算机上加载用户从文件系统中选择的图像将需要此处未介绍的其他技术。
资源在运行时加载
了解有关加载资源的第一件事是,它们当然是在运行时加载的。通常,在开发过程中,应用程序是从文件系统运行的;也就是说,运行该文件所需的类文件和资源是文件系统上的各个文件。但是,一旦构建了应用程序,通常就可以从jar文件中执行该应用程序。在这种情况下,诸如FXML文件,样式表和图像之类的资源不再是文件系统上的单个文件,而是jar文件中的条目。因此:
代码不能使用File
,FileInputStream
或file:
URL加载资源
JavaFX使用URL加载资源
JavaFX使用URL加载FXML,图像和CSS样式表。FXMLLoader
明确希望将java.net.URL
对象传递给它(传递给static
FXMLLoader.load(...)
方法,传递给FXMLLoader
构造函数或传递给setLocation()
方法)。Image
和Scene.getStylesheets().add(...)
都期望表示URL的String
。如果URL是在没有方案的情况下传递的,则它们将相对于类路径进行解释。通过在URL
上调用toExternalForm()
,可以以健壮的方式从URL
创建这些字符串。
为资源创建正确的URL的推荐机制是使用Class.getResource(...)
,在适当的Class
实例上调用它。可以通过调用getClass()
(给出当前对象的类)或ClassName.class
来获得此类实例。 Class.getResource(...)
方法采用表示资源名称的String
。
资源名称规则
资源名称是用/
分隔的路径名。每个组件代表一个包或子包名称组件。
资源名称区分大小写。
资源名称中的各个组件必须是有效的Java标识符
最后一点具有重要意义:
.
和..
不是有效的Java标识符,因此不能在资源名称中使用它们。
当应用程序从文件系统运行时,它们实际上可能起作用,尽管这确实是getResource()
实现的偶然事件。当应用程序捆绑为jar文件时,它们将失败。
同样,如果您在不能区分仅大小写不同的文件名的操作系统上运行,则从文件系统运行时在资源名称中使用错误的大小写可能会起作用,但从jar文件运行时会失败。
以/
开头的资源名称是绝对的:换句话说,它们相对于类路径进行解释。没有前导/
的资源名称将相对于调用getResource()
的类进行解释。
对此略有变化是使用getClass().getClassLoader().getResource(...)
。提供给ClassLoader.getResource(...)
的路径始终是绝对路径,即相对于类路径。
使用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资源名称,并且如果将应用程序捆绑为jar文件,则该路径将不起作用。
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
包
package org.jamesd.examples.sample.images ;
public interface ImageLocation { }
Image clubs = new Image(ImageLocation.class.getResource("clubs.png").toExternalForm());
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");
*.java
中的
src/main/java
文件被编译为类文件,这些类文件将部署到生成文件夹或jar文件中。
src/main/resources
中的资源被复制到构建文件夹或jar文件。
target/classes
中)包含单个结构。
src/main/java
和
src/main/resources
都被认为是构建中相应结构的根,因此,构建中仅包含它们的内容,而不是文件夹本身。换句话说,运行时没有
resources
文件夹可用。下面的“疑难解答”部分中显示了构建结构。
src/main/java
源文件夹与
src/main/resources
文件夹不同。在第一种情况下,它显示软件包,但对于资源文件夹,它显示文件夹。确保知道是否在IDE中创建包(名称以
.
分隔)或文件夹(名称不得包含
.
或Java标识符中无效的任何其他字符)。
.
或
..
。
Class.getResource(...)
,如果路径具有前导
/
,则该路径为绝对路径,否则为相对路径。对于
ClassLoader.getResource(...)
,该路径始终是绝对路径。
target/classes
。其他构建工具和IDE将部署到名为
bin
,
classes
,
build
或
out
的文件夹中。
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
$
关于javafx - 如何确定JavaFX应用程序所需的FXML文件,CSS文件,图像和其他资源的正确路径?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/61581762/
我是 C 语言新手,我编写了这个 C 程序,让用户输入一年中的某一天,作为返回,程序将输出月份以及该月的哪一天。该程序运行良好,但我现在想简化该程序。我知道我需要一个循环,但我不知道如何去做。这是程序
我一直在努力找出我的代码有什么问题。这个想法是创建一个小的画图程序,并有红色、绿色、蓝色和清除按钮。我有我能想到的一切让它工作,但无法弄清楚代码有什么问题。程序打开,然后立即关闭。 import ja
我想安装screen,但是接下来我应该做什么? $ brew search screen imgur-screenshot screen
我有一个在服务器端工作的 UDP 套接字应用程序。为了测试服务器端,我编写了一个简单的 python 客户端程序,它发送消息“hello world how are you”。服务器随后应接收消息,将
我有一个 shell 脚本,它运行一个 Python 程序来预处理一些数据,然后运行一个 R 程序来执行一些长时间运行的任务。我正在学习使用 Docker 并且我一直在运行 FROM r-base:l
在 Linux 中。我有一个 c 程序,它读取一个 2048 字节的文本文件作为输入。我想从 Python 脚本启动 c 程序。我希望 Python 脚本将文本字符串作为参数传递给 c 程序,而不是将
前言 最近开始整理笔记里的库存草稿,本文是 23 年 5 月创建的了(因为中途转移到 onedrive,可能还不止) 网页调起电脑程序是经常用到的场景,比如百度网盘下载,加入 QQ 群之类的 我
对于一个类,我被要求编写一个 VHDL 程序,该程序接受两个整数输入 A 和 B,并用 A+B 替换 A,用 A-B 替换 B。我编写了以下程序和测试平台。它完成了实现和行为语法检查,但它不会模拟。尽
module Algorithm where import System.Random import Data.Maybe import Data.List type Atom = String ty
我想找到两个以上数字的最小公倍数 求给定N个数的最小公倍数的C++程序 最佳答案 int lcm(int a, int b) { return (a/gcd(a,b))*b; } 对于gcd,请查看
这个程序有错误。谁能解决这个问题? Error is :TempRecord already defines a member called 'this' with the same paramete
当我运行下面的程序时,我在 str1 和 str2 中得到了垃圾值。所以 #include #include #include using namespace std; int main() {
这是我的作业: 一对刚出生的兔子(一公一母)被放在田里。兔子在一个月大时可以交配,因此在第二个月的月底,每对兔子都会生出两对新兔子,然后死去。 注:在第0个月,有0对兔子。第 1 个月,有 1 对兔子
我编写了一个程序,通过对字母使用 switch 命令将十进制字符串转换为十六进制,但是如果我使用 char,该程序无法正常工作!没有 switch 我无法处理 9 以上的数字。我希望你能理解我,因为我
我是 C++ 新手(虽然我有一些 C 语言经验)和 MySQL,我正在尝试制作一个从 MySQL 读取数据库的程序,我一直在关注这个 tutorial但当我尝试“构建”解决方案时出现错误。 (我正在使
仍然是一个初学者,只是尝试使用 swift 中的一些基本函数。 有人能告诉我这段代码有什么问题吗? import UIKit var guessInt: Int var randomNum = arc
我正在用 C++11 编写一个函数,它采用 constant1 + constant2 形式的表达式并将它们折叠起来。 constant1 和 constant2 存储在 std::string 中,
我用 C++ 编写了这段代码,使用运算符重载对 2 个矩阵进行加法和乘法运算。当我执行代码时,它会在第 57 行和第 59 行产生错误,非法结构操作(两行都出现相同的错误)。请解释我的错误。提前致谢:
我是 C++ 的初学者,我想编写一个简单的程序来交换字符串中的两个字符。 例如;我们输入这个字符串:“EXAMPLE”,我们给它交换这两个字符:“E”和“A”,输出应该类似于“AXEMPLA”。 我在
我需要以下代码的帮助: 声明 3 个 double 类型变量,每个代表三角形的三个边中的一个。 提示用户为第一面输入一个值,然后 将用户的输入设置为您创建的代表三角形第一条边的变量。 将最后 2 个步
我是一名优秀的程序员,十分优秀!