我正在从我的Java项目的编译JAR中的包中加载一个文本文件。相关目录结构如下:

/src/initialization/Lifepaths.txt

我的代码通过调用Class::getResourceAsStream来返回一个InputStream来加载一个文件。

public class Lifepaths {
    public static void execute() {
        System.out.println(Lifepaths.class.getClass().
            getResourceAsStream("/initialization/Lifepaths.txt"));
    }

    private Lifepaths() {}

    //This is temporary; will eventually be called from outside
    public static void main(String[] args) {execute();}
}

不管我用什么,输出总是输出null。我不知道为什么上面的方法不管用,所以我也尝试了一下:

“初始化/ src / / Lifepaths.txt” “初始化/ Lifepaths.txt” “Lifepaths.txt”

这两种方法都不起作用。到目前为止,我已经阅读了许多关于这个主题的问题,但没有一个是有帮助的——通常,他们只是说使用根路径加载文件,而我已经这样做了。或者只是从当前目录加载文件(只是加载文件名),我也尝试过。该文件将被编译到JAR中的适当位置,并具有适当的名称。

我怎么解决这个问题?


当前回答

不知道是否有帮助,但在我的情况下,我的资源在/src/文件夹,并得到这个错误。 然后我将图片移动到bin文件夹,它修复了这个问题。

其他回答

默认的JVM类加载器将首先使用父类加载器加载资源: .

lifepathways .class. getclass()的类加载器是引导类加载器,所以getResourceAsStream只搜索$JAVA_HOME,而不管用户提供的类路径是什么。显然,Lifepaths.txt不在那里。

Lifepaths.class的类加载器是系统类路径类加载器,所以getResourceAsStream将搜索用户定义的类路径,Lifepaths.txt在那里。

当使用java.lang.Class#getResourceAsStream(字符串名称)时,不以'/'开头的名称将作为包名的前缀添加。要避免这种情况,请使用java.lang。类加载器# getResourceAsStream代替。

例如:

ClassLoader loader = Thread.currentThread().getContextClassLoader();
String resourceName = "Lifepaths.txt";
InputStream resourceStream = loader.getResourceAsStream(resourceName); 

我发现自己也遇到了类似的问题。因为我使用maven,我需要更新我的pom.xml,包括这样的东西:

   ...
</dependencies>
<build>
    <resources>
        <resource>
            <directory>/src/main/resources</directory>
        </resource>
        <resource>
            <directory>../src/main/resources</directory>
        </resource>
    </resources>
    <pluginManagement>
        ...

请注意其中的资源标记,以指定文件夹的位置。如果你有嵌套的项目(像我一样),那么你可能想从其他领域获得资源,而不仅仅是在你工作的模块中。如果您使用类似的配置数据,这有助于减少在每次回购中保持相同的文件

如果您使用Maven,请确保您的包装是“jar”而不是“pom”。

<packaging>jar</packaging>

确保您的资源目录(例如:"src")在类路径中(确保它是eclipse中构建路径中的源目录)。

确保clazz是从主类加载器加载的。

然后,加载src/initialization/Lifepaths.txt,使用

clazz.getResourceAsStream("/initialization/Lifepaths.txt");

原因: getresourcesasstream (foo)从clazz的类路径中查找foo,相对于clazz所在的目录。前导的“/”使它从clazz类路径中的任何目录的根目录加载。

除非您在某种类型的容器中,比如Tomcat,或者直接使用classloader做一些事情,否则您可以只将eclipse/命令行类路径视为唯一的类加载器类路径。

对我有用的是将文件添加到我的项目/Java资源/src下,然后使用

this.getClass().getClassLoader().getResourceAsStream("myfile.txt");

我不需要显式地将这个文件添加到路径(添加到/src显然这样做)