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

时间:2018-10-17 18:58:40

标签: java maven intellij-idea maven-shade-plugin

我正在尝试使自己熟悉maven,并为此目的创建了一个测试项目。我创建了一个简单的类,它仅打印一些内容,并从.txt文件读取。我的主班看起来像这样:

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();
    }
}

1 个答案:

答案 0 :(得分:1)

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);
          }
       }
    }
}