- 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/61566567/
我正在通过 labrepl 工作,我看到了一些遵循此模式的代码: ;; Pattern (apply #(apply f %&) coll) ;; Concrete example user=> (a
我从未向应用商店提交过应用,但我会在不久的将来提交。 到目前为止,我对为 iPhone 而非 iPad 进行设计感到很自在。 我了解,通过将通用PAID 应用放到应用商店,客户只需支付一次就可以同时使
我有一个应用程序,它使用不同的 Facebook 应用程序(2 个不同的 AppID)在 Facebook 上发布并显示它是“通过 iPhone”/“通过 iPad”。 当 Facebook 应用程序
我有一个要求,我们必须通过将网站源文件保存在本地 iOS 应用程序中来在 iOS 应用程序 Webview 中运行网站。 Angular 需要服务器来运行应用程序,但由于我们将文件保存在本地,我们无法
所以我有一个单页客户端应用程序。 正常流程: 应用程序 -> OAuth2 服务器 -> 应用程序 我们有自己的 OAuth2 服务器,因此人们可以登录应用程序并获取与用户实体关联的 access_t
假设我有一个安装在用户设备上的 Android 应用程序 A,我的应用程序有一个 AppWidget,我们可以让其他 Android 开发人员在其中以每次安装成本为基础发布他们的应用程序推广广告。因此
Secrets of the JavaScript Ninja中有一个例子它提供了以下代码来绕过 JavaScript 的 Math.min() 函数,该函数需要一个可变长度列表。 Example:
当我分别将数组和对象传递给 function.apply() 时,我得到 NaN 的 o/p,但是当我传递对象和数组时,我得到一个数字。为什么会发生这种情况? 由于数组也被视为对象,为什么我无法使用它
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界. 这篇CFSDN的博客文章ASP转换格林威治时间函数DateDiff()应用由作者收集整理,如果你
我正在将列表传递给 map并且想要返回一个带有合并名称的 data.frame 对象。 例如: library(tidyverse) library(broom) mtcars %>% spl
我有一个非常基本的问题,但我不知道如何实现它:我有一个返回数据框,其中每个工具的返回值是按行排列的: tmp<-as.data.frame(t(data.frame(a=rnorm(250,0,1)
我正在使用我的 FB 应用创建群组并邀请用户加入我的应用群组,第一次一切正常。当我尝试创建另一个组时,出现以下错误: {"(OAuthException - #4009) (#4009) 在有更多用户
我们正在开发一款类似于“会说话的本”应用程序的 child 应用程序。它包含大量用于交互式动画的 JPEG 图像序列。 问题是动画在 iPad Air 上播放正常,但在 iPad 2 上播放缓慢或滞后
我关注 clojure 一段时间了,它的一些功能非常令人兴奋(持久数据结构、函数式方法、不可变状态)。然而,由于我仍在学习,我想了解如何在实际场景中应用,证明其好处,然后演化并应用于更复杂的问题。即,
我开发了一个仅使用挪威语的应用程序。该应用程序不使用本地化,因为它应该仅以一种语言(挪威语)显示。但是,我已在 Info.plist 文件中将“本地化 native 开发区域”设置为“no”。我还使用
读完 Anthony's response 后上a style-related parser question ,我试图说服自己编写单体解析器仍然可以相当紧凑。 所以而不是 reference ::
multicore 库中是否有类似 sapply 的东西?还是我必须 unlist(mclapply(..)) 才能实现这一点? 如果它不存在:推理是什么? 提前致谢,如果这是一个愚蠢的问题,我们深表
我喜欢在窗口中弹出结果,以便更容易查看和查找(例如,它们不会随着控制台继续滚动而丢失)。一种方法是使用 sink() 和 file.show()。例如: y <- rnorm(100); x <- r
我有一个如下所示的 spring mvc Controller @RequestMapping(value="/new", method=RequestMethod.POST) public Stri
我正在阅读 StructureMap关于依赖注入(inject),首先有两部分初始化映射,具体类类型的接口(interface),另一部分只是实例化(请求实例)。 第一部分需要配置和设置,这是在 Bo
我是一名优秀的程序员,十分优秀!