使用可执行Jar时找不到Spring-Boot资源

Mic*_*ner 5 java resources filenotfoundexception executable-jar spring-boot

我再次遇到一个奇怪的问题,希望有人可以提供帮助.

我有一个spring boot后端模块,在Eclipse中启动main时,应用程序是可执行的.一切都很好.

我的应用程序使用ss/main/resources文件夹中包含的csv文件将示例数据导入数据库.如上所述,在eclipse中启动时一切正常.

现在我想将它作为可执行jar执行,应用程序开始启动然后无法启动,因为它无法找到csv文件.它打印出来的路径,它在哪里查找文件,是正确的,并且csv文件包含在jar中.

该模块的Pom如下所示:

<project>
    <modelVersion>4.0.0</modelVersion>

    <parent>
        <groupId>at.company.bbsng</groupId>
        <artifactId>bbsng-import</artifactId>
        <version>0.1.0-SNAPSHOT</version>
    </parent>

    <artifactId>bbsng-import-backend</artifactId>
    <name>bbsng-import-backend</name>

    <properties>
        <start-class>at.company.bbsng.dataimport.Application</start-class>
    </properties>


    <dependencies>

        <!-- SPRING ... -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-batch</artifactId>
            <!-- EXCLUDE LOGBACK AND USE LOG4J -->
            <exclusions>
                <exclusion>
                    <artifactId>spring-boot-starter-logging</artifactId>
                    <groupId>org.springframework.boot</groupId>
                </exclusion>
            </exclusions>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-log4j</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>

        <!-- COMMONS ... -->

        ...

    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
                <executions>
                    <execution>
                        <goals>
                            <goal>repackage</goal>
                        </goals>
                    </execution>
                </executions>
            </plugin>
        </plugins>
    </build>

</project>
Run Code Online (Sandbox Code Playgroud)

csv文件的路径在propery文件中配置如下:

# EXAMPLE PATH
csv.path=config/csv/
Run Code Online (Sandbox Code Playgroud)

java配置文件的部分如下:

  ...

  @Value("${csv.path}")
  private String csvExamplePath;

  @Bean
  public Resource addressResource() {
    return new ClassPathResource(csvExamplePath + CSV_ADDRESS);
  }

    ...
Run Code Online (Sandbox Code Playgroud)

在jar中,文件位于路径中

\config\csv\
Run Code Online (Sandbox Code Playgroud)

堆栈跟踪:

Caused by: java.io.FileNotFoundException: class path resource [config/csv/Company.csv] cannot be resolved to absolute file path because it does not reside in th
e file system: jar:file:/C:/Development/Projekte/bbsng/trunk/import/backend/target/bbsng-import-backend-0.1.0-SNAPSHOT.jar!/config/csv/Company.csv
        at org.springframework.util.ResourceUtils.getFile(ResourceUtils.java:207)
        at org.springframework.core.io.AbstractFileResolvingResource.getFile(AbstractFileResolvingResource.java:52)
        at at.compax.bbsng.dataimport.app.source.company.CompanyGenerator.init(CompanyGenerator.java:28)
        at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
        at sun.reflect.NativeMethodAccessorImpl.invoke(Unknown Source)
        at sun.reflect.DelegatingMethodAccessorImpl.invoke(Unknown Source)
        at java.lang.reflect.Method.invoke(Unknown Source)
        at org.springframework.beans.factory.annotation.InitDestroyAnnotationBeanPostProcessor$LifecycleElement.invoke(InitDestroyAnnotationBeanPostProcessor.java
Run Code Online (Sandbox Code Playgroud)

同样,应用程序在从eclipse启动时按预期工作,只有可执行jar抱怨缺少csv文件,jar中已有的内容.

任何线索都会很棒.

Mic*_*ner 15

好的,我已经找到了真正的问题和解决方案.

首先,应用程序使用正确的csv文件路径,但是在使用我在以下链接中找到的可执行jar时还有另一个问题.计算器,链接

在我发布可执行jar之前,我使用以下解决方案来获取CSV文件(问题是getFile()):

final List<String> resourceLines = FileReadUtils.readLines(specialisationResource.getFile());
for (final String line : resourceLines) {
  data.add(getNewTransientSpecialisation(line));
}
Run Code Online (Sandbox Code Playgroud)

但是在可执行的jar中我无法将我的资源用作文件,我需要将其用作流,请参阅上面提供的链接.所以我需要更改我的代码.如果您更喜欢使用本机java,则可以执行以下操作:

final InputStream inputStream = specialisationResource.getInputStream();
final BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(inputStream));

String line;
while ((line = bufferedReader.readLine()) != null) {
    data.add(getNewTransientSpecialisation(line));
}
Run Code Online (Sandbox Code Playgroud)

我更喜欢使用框架并使用如下的apache公共:

final List<String> resourceLines = IOUtils.readLines(specialisationResource.getInputStream());
for (final String line : resourceLines) {
    data.add(getNewTransientSpecialisation(line));
}
Run Code Online (Sandbox Code Playgroud)

所以请记住,不要使用File()来获取资源,总是使用流从开始时避免该问题:-)

希望能帮助别人.