zoukankan      html  css  js  c++  java
  • SpringBoot读取war包jar包Resource资源文件解决办法

    SpringBoot读取war包jar包Resource资源文件解决办法

    场景描述

    在开发过程中我们经常会碰到要在代码中获取资源文件的情况,而我在最近在SpringBoot项目中时碰到一个问题,就是在本地运行时,获取本地的xml资源文件是能够获取到的,但是项目打成war包jar包启动运行时,就会发生问题,报找不到资源文件的错误。然后经过寻找排查确定了是下面代码通过ClassLoader获取路径的时候出错了。
     
    常用方式:
     
    /**
     * @author mazhq
     * @Title: TestMain
     * @ProjectName: zeus
     * @Description: TODO
     * @date 2019/3/5 16:10
     */
    public class TestMain {
        public static void main(String[] args) {
            String path = TestMain.class.getClassLoader().getResource("1.xml").getPath();
            System.out.println(path);
        }
     /**
         * 输出:
         * 
         */D:/demo_projects/sc-architecture/service-hi/target/classes/1.xml
         */
    }
    

      

    但是在将SpringBoot打包放到Linux服务器启动打印的目录为

    /data/zeus/service-hi-1.0.0-SNAPSHOT.war!/WEB-INF/classes!/1.xml
    

    可以看到在Linux中无法直接访问未经解压的文件,所以就会找不到文件。

    解决办法

      1. 通过ClassLoadergetResourceAsStream()方法获取其流,就能够获取到。

      读取jar里面的文件,我们只能用流去读取,不能用File

    public class TestMain {
        public static void main(String[] args) {
            try {
                List<String> content = IOUtils.readLines(TestMain.class.getClassLoader().getResourceAsStream("1.xml"), "UTF-8");
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }
    

      2. 采用绝对路径将文件放到服务器某个路径,在application.properties中配置路径读取。

      3. 不推荐:将内容放到数据库中。

    获取资源的两种方式

    通常在开发过程中会碰到读取配置文件的问题,一般有两种方式进行读取。一种是Class.getResource(String path),一种是ClassLoader.getResource(String path),这两种虽然都能读取文件,但是在path的填写上有一点点的不同。

    Class.getResource

    1. path以/开头:则是从ClassPath根下获取
    2. path不以/开头:默认是从此类所在的包下取资源

    下面有个例子

    public class TestMain {
        public static void main(String[] args) {
            System.out.println(TestMain.class.getResource("/"));
            System.out.println(TestMain.class.getResource(""));
        }
        /**
         * 输出:
         * 
         * file:/D:/demo_projects/sc-architecture/service-hi/target/classes/
         * file:/D:/demo_projects/sc-architecture/service-hi/target/classes/com/mazhq/servicehi/
         */
    }
    

      

    那么读取在resource下的1.xml,就如下的获取方法

    public class TestMain {
        public static void main(String[] args) {
            System.out.println(TestMain.class.getResource("/1.xml"));
            System.out.println(TestMain.class.getResource("../../../1.xml"));
        }
        /**
         * 输出:
         *
         * file:/D:/demo_projects/sc-architecture/service-hi/target/classes/1.xml
         * file:/D:/demo_projects/sc-architecture/service-hi/target/classes/1.xml
         */
    }

    ClassLoader.getResource

    ClassLoader.getResource的path中不能以/开头,path是默认是从根目录下进行读取的

    代码如下:

    public class TestMain {
        public static void main(String[] args) {
            System.out.println(TestMain.class.getClassLoader().getResource(""));
            System.out.println(TestMain.class.getClassLoader().getResource("/"));
        }
        /**
         * 输出:
         *
         * file:/D:/demo_projects/sc-architecture/service-hi/target/classes/
         * null
         */
    }
    

      

    从上面例子我们可以看到

    TestMain.class.getClassLoader().getResource("")=TestMain.class.getResource("/")

    两个获取资源文件的差别

    其实查看Class.getResource中可以看到

    public java.net.URL getResource(String name) {
            name = resolveName(name);
            ClassLoader cl = getClassLoader0();
            if (cl==null) {
                // A system class.
                return ClassLoader.getSystemResource(name);
            }
            return cl.getResource(name);
        }
    

    他最后调用的还是ClassLoader.getResource这个方法,那么为什么会有path的差别呢,因为其resolveName方法中对传的/进行了解析,解析为了空字符串。

    resolveName 方法实现如下:

     private String resolveName(String name) {
            if (name == null) {
                return name;
            }
            if (!name.startsWith("/")) {
                Class<?> c = this;
                while (c.isArray()) {
                    c = c.getComponentType();
                }
                String baseName = c.getName();
                int index = baseName.lastIndexOf('.');
                if (index != -1) {
                    name = baseName.substring(0, index).replace('.', '/')
                        +"/"+name;
                }
            } else {
                name = name.substring(1);
            }
            return name;
        }

      //传入 "/" 返回  ""

      

    最后:大家用的时候注意一下这些问题,避免在这个上面耽误时间。

  • 相关阅读:
    LNMP 部署
    zabbix3.2安装graphtree3.0.4
    升级java8---from centos
    mysql5.6-5.7性能调优
    samba server install
    centos7 zabbix3 install done
    实验四总结
    第五周学习小结
    个人的一些html、css笔记
    为什么wait,notify,notifyAll定义在Object中?
  • 原文地址:https://www.cnblogs.com/owenma/p/10477910.html
Copyright © 2011-2022 走看看