我再次面临一个奇怪的问题,希望这里有人能提供帮助。
我有一个spring boot后端模块,在eclipse中可以正常工作,在main.application中启动main时,应用程序是可执行的。一切都很好。
我的应用程序使用csv文件将示例数据导入数据库,其中包含src / main / resources文件夹中的内容。如前所述,在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>
CSV文件的路径在属性文件中配置如下:
# EXAMPLE PATH csv.path=config/csv/
Java配置文件的部分如下:
... @Value("${csv.path}") private String csvExamplePath; @Bean public Resource addressResource() { return new ClassPathResource(csvExamplePath + CSV_ADDRESS); } ...
在jar中,文件位于路径
\config\csv\
堆栈跟踪:
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
同样,从Eclipse启动应用程序时,它可以按预期工作,只有可执行的jar抱怨缺少csv文件,而jar中已经有什么。
任何线索都很好。
好的,我已经找到了真正的问题和解决方案。
首先,应用程序使用了正确的csv文件路径,但是在使用可执行jar时,还有另一个问题。
在我开始讨论可执行jar之前,我使用以下解决方案来获取CSV文件(问题是getFile()):
final List<String> resourceLines = FileReadUtils.readLines(specialisationResource.getFile()); for (final String line : resourceLines) { data.add(getNewTransientSpecialisation(line)); }
但是在可执行的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)); }
我更喜欢使用框架并使用如下所示的apache commons:
final List<String> resourceLines = IOUtils.readLines(specialisationResource.getInputStream()); for (final String line : resourceLines) { data.add(getNewTransientSpecialisation(line)); }
因此,请记住,不要使用File()获取资源,请始终使用流来避免从一开始就出现该问题:-)
希望能对某人有所帮助。