在可执行 jar 中使用资源文件

Using resource files in an executeable jar

我正在尝试熟悉 Maven 并为此创建了一个测试项目。我创建了一个简单的 class,它只打印一些内容,从 .txt 文件中读取。我的主要 class 看起来像这样:

public class HelloWorld {
    public static void main(String[] args) throws IOException {
        String filePath = HelloWorld.class.getClassLoader().getResource("test.txt").getFile();

        BufferedReader br = new BufferedReader(new FileReader(filePath));
        String line;
        while ((line = br.readLine()) != null) {
            System.out.println(line);
        }
        br.close();
    }
}

我创建了一个资源文件夹,将其标记为资源根目录,修改了资源模式并从我的项目中打包了一个可执行的 jar。我的项目结构如下所示:

Quickstart
├───.idea
├───src
│   ├───main
│   │   ├───java
│   │   │   └───de
│   │   │       └───mb
│   │   │           └───hello
|   |   |               └───HelloWorld.java
│   │   └───resources
|   |       └───test.txt

现在我的问题是,当我尝试执行我的 jar 时,出现以下错误:

Exception in thread "main" java.io.FileNotFoundException:   
file:\C:\Users\mb\IdeaProjects\Quickstart\target\Quickstart-1.
0-SNAPSHOT.jar!\test.txt (The filename, directory name, or volume label syntax is incorrect)
    at java.io.FileInputStream.open0(Native Method)
    at java.io.FileInputStream.open(FileInputStream.java:195)
    at java.io.FileInputStream.<init>(FileInputStream.java:138)
    at java.io.FileInputStream.<init>(FileInputStream.java:93)
    at java.io.FileReader.<init>(FileReader.java:58)
    at de.mb.hello.HelloWorld.main(HelloWorld.java:15)

我想问题是,.txt 文件在 .jar 中,但我应该如何声明路径才能使其工作?我已经尝试过使用不同的方法让我的资源路径无济于事并修改了我的 pom.xml:

<project>
    ...
    <build>
        <resources>
            <resource>
            <directory>src/main/resources</directory>
            <includes>
                <include>**/*.txt</include>
            </includes>
            </resource>
        </resources>
        ...
        <pluginManagement>
            <plugins>
                <plugin>
                    <groupId>org.apache.maven.plugins</groupId>
                    <artifactId>maven-shade-plugin</artifactId>
                    <version>3.2.0</version>
                    <configuration>
                    <transformers>
                        <transformer implementation="org.apache.maven.plugins.shade.resource.ManifestResourceTransformer">
                        <manifestEntries>
                            <Main-Class>de.mb.hello.HelloWorld</Main-Class>
                        </manifestEntries>
                        </transformer>
                    </transformers>
                    </configuration>
                    <executions>
                    <execution>
                        <phase>package</phase>
                        <goals>
                            <goal>shade</goal>
                        </goals>
                    </execution>
                    </executions>
                </plugin>
            </plugins>
        </pluginManagement>
    </build>
</project>

更新: 按照建议,我更改了我的代码以使用 getResourceAsStream() 而不是 getResource() 并且它有效,如下所示:

public static void main(String[] args) {
    try {
        InputStream in = HelloWorld.class.getClassLoader().getResourceAsStream("test.txt");
        BufferedReader br = new BufferedReader(new InputStreamReader(in));
        try {
            String line;
            while ((line = br.readLine()) != null) {
                System.out.println(line);
            }
        } finally {
            in.close();
            br.close();
        }
    } catch (Exception e) {
        e.printStackTrace();
    }
}

你的jar文件中打包的资源不是一个文件而是一个zip文件中的一系列字节。它必须处理有一个字节流。

使用 getResourceAsStream(...) 而不是 getResource(...) 来获取 InputStream 而不是 File 并使用 InputStreamReader 而不是 FileReader 读取内容。

不要忘记在 finally 块中或使用 try-with-resources.

关闭资源

类似的东西:

public class HelloWorld {
    public static void main(String[] args) throws IOException {
        try(BufferedReader br = new BufferedReader(new InputStreamReader(HelloWorld.class.getClassLoader().getResourceAsStream("test.txt")))) {
          String line;
          while ((line = br.readLine()) != null) {
              System.out.println(line);
          }
       }
    }
}