温馨提示×

温馨提示×

您好,登录后才能下订单哦!

密码登录×
登录注册×
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》

项目打包成jar后包无法读取src/main/resources下文件怎么解决

发布时间:2022-04-02 13:42:55 来源:亿速云 阅读:1166 作者:iii 栏目:开发技术

这篇文章主要介绍“项目打包成jar后包无法读取src/main/resources下文件怎么解决”,在日常操作中,相信很多人在项目打包成jar后包无法读取src/main/resources下文件怎么解决问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”项目打包成jar后包无法读取src/main/resources下文件怎么解决”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!

    一、项目场景

    在项目中读取文件时, 使用new File() 出现的一个坑以及解决流程
    这种问题不仅在本地文件读取时会遇到, 而且在下载项目下 (例如: src/main/resources目录下) 的文本时, 也会遇到,

    二、问题描述

    发现问题

    原来代码
    该代码功能是利用 common.io 包下的FileUtils来读取文件, 放到一个字符串中

    String s = FileUtils.readFileToString(new File("src/main/resources/holiday.txt"), "utf-8");

    这种路径书写方式 new File("src/main/resources/holiday.txt") , 在本地运行没问题,
    但是打包之后在服务器中运行出现了问题. 下面是错误截图

    项目打包成jar后包无法读取src/main/resources下文件怎么解决

    可以看到在服务器中日志提示: java.io.FileNotFoundException: File 'holiday.txt' does not exist
    即: 在打包后, 一开始配置的路径src/main/resources下无法找到该文件

    分析问题

    项目在打包之后, 位于 resource目录下的文件, 最常见的就是各种Spring配置文件就会打包在 BOOT-INF/classes 目录下
    而FIle 在按照原来的文件路径src/main/resources/holiday.txt'去寻找, 必然找不到文件, 因此会报文件找不到的异常

    项目打包成jar后包无法读取src/main/resources下文件怎么解决

    在定位问题的过程中发现, 这里 提供了一个思路
    就是SpringBoot中所有文件都在jar包中,没有一个实际的路径,因此可以使用以下方式

        /**
         * 通过ClassPathResource类获取,建议SpringBoot中使用
         * springboot项目中需要使用此种方法,因为jar包中没有一个实际的路径存放文件
         *
         * @param fileName
         * @throws IOException
         */
        public void function6(String fileName) throws IOException {
            ClassPathResource classPathResource = new ClassPathResource(fileName);
            InputStream inputStream = classPathResource.getInputStream();
            getFileContent(inputStream);
        }

    为什么使用 ClassPathResource 后, 可以找到打包后的文件路径?

    上面代码的核心就是: 实例化ClassPathResource 对象. 然后调用getInputStream 来获取资源文件

    下面我们来分析这些代码
    ClassPathResource 在实例化时, 会初始化类加载器 classLoader 并将项目所用到的所有路径加载到类加载器 classLoader 中, 这些路径包括: java运行环境的jar, Maven 项目中的jar, 以及当前项目打包后的jar等(如下图)

    项目打包成jar后包无法读取src/main/resources下文件怎么解决

    classPathResource.getInputStream 在获取资源文件时, 因为上面我们初始化了一个classLoader.
    所以classLoader不为空, 因此会执行 getResourceAsStream 方法, 我们来追一下这个方法

    项目打包成jar后包无法读取src/main/resources下文件怎么解决

    getResourceAsStream 方法中的getResource是实际的业务处理方法, 我们继续深入

    项目打包成jar后包无法读取src/main/resources下文件怎么解决

    getResource 方法如下图, 实际的功能就是递归调用自己, 去不断遍历 parent 下的路径, 获取对应的资源文件
    那么 parent 又是谁呢? 我们继续往下看

    项目打包成jar后包无法读取src/main/resources下文件怎么解决

    看到这里我们豁然开朗, 这个神秘的 parent 就是类加载器classLoader!!!
    因此getResource 方法就是去不断遍历我们在ClassPathResource实例化时, 创建的类加载器下面的路径!!!(对应第1点)

    项目打包成jar后包无法读取src/main/resources下文件怎么解决

    三、解决方案

    原来读取文件的代码如下

    String s = FileUtils.readFileToString(new File("src/main/resources/holiday.txt"), "utf-8");

    去查看 File 的构造函数, 看能否通过 InputStream 来构造
    从下图看是不行的

    项目打包成jar后包无法读取src/main/resources下文件怎么解决

    方案一

    并且我们发现 org.apache.commons.io没有提供ClassPathResource 作为入参的读取文件的方法.
    因此我们必须手写读取文件的方法

    项目打包成jar后包无法读取src/main/resources下文件怎么解决

    手写的代码如下
    主要注意 Resource resource = new ClassPathResource(fileName); is = resource.getInputStream();

        /**
         * Java读取txt文件的内容
         *
         * @param fileName resources目录下文件名称(无需带目录)
         * @return 将每行作为一个单位放到list中
         */
        public static List<String> readTxtFile(String fileName) {
            List<String> listContent = new ArrayList<>();
            InputStream is = null;
            InputStreamReader isr = null;
            BufferedReader br = null;
            String encoding = "utf-8";
            try {
                Resource resource = new ClassPathResource(fileName);
                is = resource.getInputStream();
                isr = new InputStreamReader(is, encoding);
                br = new BufferedReader(isr);
                String lineTxt = null;
                while ((lineTxt = br.readLine()) != null) {
                    listContent.add(lineTxt);
                }
            } catch (IOException e) {
                e.printStackTrace();
            } finally {
                try {
                    br.close();
                    isr.close();
                    is.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            return listContent;
        }

    方案二

    这种方式对代码入侵较小, 核心还是利用 common.io 下的 FileUtils, 具体方法是
    利用FileUtils将ClassPathResource.getInputStream 得到的输入流复制到临时文件中, 然后读取这个临时文件
    这种方式缺点是: 需要创建临时文件, 如果待读取文件过大, 则重新创建文件和复制操作会消耗一定的空间和时间, 影响性能

      //方式二 利用FileUtils将ClassPathResource.getInputStream 得到的输入流复制到临时文件中
      Resource resource = new ClassPathResource("holiday.txt");
      InputStream inputStream = resource.getInputStream();
      File tempFile = File.createTempFile("temp", ".txt");
      FileUtils.copyInputStreamToFile(inputStream, tempFile);
      
      String s = FileUtils.readFileToString(tempFile, StandardCharsets.UTF_8);

    意外出现

    到这里又出现了一个问题, 就是我用的测试项目因为在 maven 里面指定了某些格式的文件. 如下配置
    因为指定了banner.txt 以及 xml 与 properties结尾的文件作为资源被打包. 所以文件 holiday.txt 运行后还是访问不到
    有问题的pom.xml文件如下

    	<!-- 资源拷贝插件,实现在打包时自动拷贝java目录下以及resources目录下的xml的配置文件 -->
    		<resources>
    			<resource>
    				<directory>src/main/java</directory>
    				<includes>
    					<include>**/*.xml</include>
    				</includes>
    			</resource>
    			<resource>
    				<directory>src/main/resources</directory>
    				<includes>
    					<include>**/*.xml</include>
    					<include>**/*.properties</include>
    					<include>**/banner.txt</include>
    				</includes>
    			</resource>
    		</resources>

    打包后资源文件截图如下, 从该图中可以看到 holiday.txt 没有被打包进来

    项目打包成jar后包无法读取src/main/resources下文件怎么解决

    程序运行之后的错误截图

    项目打包成jar后包无法读取src/main/resources下文件怎么解决

    我们修改下指定打包的配置 <include>**/*.txt</include>
    这样配置后, 我们就可以将类路径下的所有txt 文件打包进行项目中了, 打包之后文件位置如下图
    或者我们可以去除项目中下面的代码配置, 这样做会默认打包 resources 下面的所有文件

    	<!-- 资源拷贝插件,实现在打包时自动拷贝java目录下以及resources目录下的xml的配置文件 -->
    		<resources>
    			<resource>
    				<directory>src/main/java</directory>
    				<includes>
    					<include>**/*.xml</include>
    				</includes>
    			</resource>
    			<resource>
    				<directory>src/main/resources</directory>
    				<includes>
    					<include>**/*.xml</include>
    					<include>**/*.properties</include>
    					<include>**/*.txt</include>
    				</includes>
    			</resource>
    		</resources>

    修改pom文件后, 重新打包后资源文件(从这里可以看到 holiday.txt 被打包进来 )

    项目打包成jar后包无法读取src/main/resources下文件怎么解决

    到此,关于“项目打包成jar后包无法读取src/main/resources下文件怎么解决”的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注亿速云网站,小编会继续努力为大家带来更多实用的文章!

    向AI问一下细节

    免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

    jar
    AI