Java 优化:读取配置文件 "万能方式" 跨平台,动态获取文件的绝对路径

每博一文案

  1. 往事不会像烟雾似的飘散,将永远像铅一般沉重地浇铸在心灵的深处。
  2. 不过,日常生活的纷繁不会让人专注地沉湎于自己的痛苦
  3. 不幸,即使人的心灵伤痕累累,也还得要去为现实中的生存和发展而挣扎。
  4. —————— 《平凡世界》
  5. 每个人的生活同样也是一个世界,即使最平凡的人,也得要为他那个世界的存在而战斗。从这个意义
  6. 上说,在这些平凡的世界里,也没有一天是平静的。
  7. —————— 《平凡世界》

@


我们知道在 Java 中读取一些配置文件信息,是在开发中十分常用的要求。

例如:这里我们使用 JDBC 实例:连接MySQL 数据库,读取连接数据库的 用户名,密码

如下是一个名为 jdbc.properties 的配置文件信息,以及存在目录



  1. package blogs.blogs8;
  2. import java.io.FileInputStream;
  3. import java.io.IOException;
  4. import java.util.Properties;
  5. public class IORead {
  6. public static void main(String[] args) {
  7. FileInputStream f = null;
  8. try {
  9. // 创建字节输入流对象
  10. // 在IDEA 中的默认相对路径是在 src 同级目录下的
  11. f = new FileInputStream("src/blogs/blogs8/jdbc.properties");
  12. // 创建Map集合中的 Properties 对象
  13. Properties properties = new Properties();
  14. properties.load(f);
  15. // 通过 key 读取对应的键值对
  16. String user = properties.getProperty("user");
  17. System.out.println(user);
  18. String password = properties.getProperty("password");
  19. System.out.println(password);
  20. } catch (IOException e) {
  21. e.printStackTrace();
  22. } finally {
  23. // 关闭IO资源
  24. if(f == null) {
  25. try {
  26. f.close();
  27. } catch (IOException e) {
  28. e.printStackTrace();
  29. }
  30. }
  31. }
  32. }
  33. }

说明:

上述的读取文件的方式,我们可以看到是 “完全没有问题的” 可以读取到对应的配置信息,但是存在一个缺点:就是移除性差。src 中是在 IDEA 这个编译器中体现的,如果是在其它的编译器中运行的时候,很大的可能会报错,原因是:这里我们使用的相对路径是,在 IDEA中的,IDEA 中的默认相对路径是 在 project 下的也就是 src 的同级目录。但是其它的系统,或者编译器就可能不是这个和 IDEA 中默认相对路径了。运行程序时,就有可能会报错:如下:找不到指定的文件。

上述这种方式:如果我们不写相对路径,而是写绝对路径的话,也是存在一个问题的。那就是因为该绝对路径是写死了的,不是动态获取的,该路径在 Windows 操作系统中是存在盘符的,所以写绝对路径的时候是需要带上盘符(E盘,D盘的),但是如果该程序是运行在其他操作系统中的话,比如 Linux 操作系统中是没有盘符的说法的。所以就会出问题。无法跨平台。

1. 优化方式一:返回一个文件的绝对路径

接下来说一种比较通用的一种路径:即使代码换位置了,这样的代码编写的方式仍然是通用的。因为该文件的路径是动态获取的。

在Windows中的话,就以该系统的文件规则,动态获取到的绝对路径是带盘符的,而 Linux系统中就以该系统的文件规则,获取到的绝对路径是不带盘符的。 这就可以跨平台了。

注意: 使用该方式的前提是:所读取的文件必须是在 类路径 下才行。如果不是在类路径下,运行程序时是会报错:系统找不到指定的路径

什么是类路径 ?

类路径也是一种特殊的相对路径,只不过它相对的是class文件。在 IDEA 中的类路径是在 src 目录下的。重点记住它

该方式的核心代码:

  1. String path = Thread.currentThread().getContextClassLoader().getResource("db.properties").getPath();
  2. /*
  3. 解释:
  4. Thread.currentThread() 当前线程对象
  5. getContextClassLoader() 是线程对象的方法,可以获取到当前线程的类加载对象
  6. getResource() 获取资源:这是类加载器对象的方法,当前线程的类加载器默认从类的根路径下加载资源。
  7. getPath() 获取当文件的绝对路径
  8. */

1.1 情况一

所读取的文件是直接存放在 src 的目录下的,该文件的并没有其它的的包。如下图所示:可以直接写文件名 + 文件名的后缀即可。

  1. import java.io.FileInputStream;
  2. import java.io.IOException;
  3. import java.util.Properties;
  4. public class IORead {
  5. public static void main(String[] args) {
  6. String path = Thread.currentThread().getContextClassLoader().getResource("db.properties").getPath();
  7. System.out.println(path); // 返回该文件的绝对路径:
  8. }
  9. }

通过该方式获取到指定文件的绝对路径,再将该绝对路径,作为参数,创建FileInputStream字节输入流对象


  1. import java.io.FileInputStream;
  2. import java.io.IOException;
  3. import java.util.Properties;
  4. public class IORead {
  5. public static void main(String[] args) {
  6. FileInputStream f = null;
  7. try {
  8. // 获取到该配置文件的的绝对路径
  9. String path = Thread.currentThread().getContextClassLoader().getResource("db.properties").getPath();
  10. // 通过该获取的文件的绝对路径创建 字节输入流对象
  11. f = new FileInputStream(path);
  12. // 创建Map集合中的 Properties 对象
  13. Properties properties = new Properties();
  14. properties.load(f);
  15. // 通过 key 读取对应的键值对
  16. String user = properties.getProperty("user");
  17. System.out.println(user);
  18. String password = properties.getProperty("password");
  19. System.out.println(password);
  20. } catch (IOException e) {
  21. e.printStackTrace();
  22. } finally {
  23. // 关闭IO资源
  24. if (f != null) {
  25. try {
  26. f.close();
  27. } catch (IOException e) {
  28. e.printStackTrace();
  29. }
  30. }
  31. }
  32. }
  33. }

也是可以读取到文件中是在 src 目录下。

1.2 情况二

当所读取的文件,是在 src 目录下,但是该 src 目录下还有其他的包(目录),则不可以直接写 “文件名+ 文件后缀名”了,而是需要写明该 src 包(目录)下的 相对路径:如下图所示的文件:该路径名应该是:blogs/blogs8/jdbc.properties

举例:


  1. import java.io.FileInputStream;
  2. import java.io.IOException;
  3. import java.util.Properties;
  4. public class IORead {
  5. public static void main(String[] args) {
  6. FileInputStream f = null;
  7. try {
  8. // 获取到该配置文件的的绝对路径,如下src目录下还有目录(包),需要指定 src目录下/包下的哪个文件。
  9. String path = Thread.currentThread().getContextClassLoader().getResource("blogs/blogs8/jdbc.properties").getPath();
  10. // 通过该获取的文件的绝对路径创建 字节输入流对象
  11. f = new FileInputStream(path);
  12. // 创建Map集合中的 Properties 对象
  13. Properties properties = new Properties();
  14. properties.load(f);
  15. // 通过 key 读取对应的键值对
  16. String user = properties.getProperty("user");
  17. System.out.println(user);
  18. String password = properties.getProperty("password");
  19. System.out.println(password);
  20. } catch (IOException e) {
  21. e.printStackTrace();
  22. } finally {
  23. // 关闭IO资源
  24. if (f != null) {
  25. try {
  26. f.close();
  27. } catch (IOException e) {
  28. e.printStackTrace();
  29. }
  30. }
  31. }
  32. }

2. 优化方式二:返回一个 InputStream 字节输入流

上述方式一:我们需要通过 :new 一个 FileInputStream 字节输入流对象的方式,这里我们直接通过指定的文件名的,直接返回一个 InputStream 字节输入流 ,不需要 new 。

同样的:该读取的文件必须是在类路径下才行,这里的IDEA的类路径是 src 目录下

核心代码如下:

  1. // 直接以流的形式返回。
  2. InputStream resourceAsStream = Thread.currentThread().getContextClassLoader().
  3. getResourceAsStream("db.properties");

举例:


  1. import java.io.IOException;
  2. import java.io.InputStream;
  3. import java.util.Properties;
  4. public class IORead {
  5. public static void main(String[] args) {
  6. // 直接在 src目录下没有包含任何子目录,可以直接写文件名+ 后缀,而如果有子目录,需要指明子目录下的文件名+后缀名
  7. InputStream inputStream = Thread.currentThread().getContextClassLoader().getResourceAsStream("db.properties");
  8. // 创建 Properties 集合对象,通过流获取指定配置文件中的键值对信息
  9. Properties properties = new Properties();
  10. try {
  11. properties.load(inputStream);
  12. } catch (IOException e) {
  13. e.printStackTrace();
  14. }
  15. String user = properties.getProperty("user");
  16. System.out.println(user);
  17. String password = properties.getProperty("password");
  18. System.out.println(password);
  19. // 关闭IO资源
  20. if (inputStream != null) {
  21. try {
  22. inputStream.close();
  23. } catch (IOException e) {
  24. e.printStackTrace();
  25. }
  26. }
  27. }
  28. }

3. 优化方式三:java.util 包下提供了一个资源绑定器

上述两个方式可以获取到任意文件的信息。

但是以下这个方式三:就只能获取到 类路径下的以 .properties 后缀的配置文件信息了。

java.util 包下提供了一个资源绑定器,便于获取属性.properties 配置文件中的内容。

该资源绑定器:只能绑定 xxx.properties 配置文件 ,并且这个文件必须在 类路径下,这里的 IDEA 是 src 目录下。

并且在写路径的时候,路径后面的扩展名不能写,写了会报错: ``。因为既然只能读取 properteis 后缀的文件,那就不用再多余的写文件后缀名了。

如果在 src 目录下的子目录中的文件,需要指明是 src 下的哪个子目录下的文件,同样不要写文件后缀名,不然报错。

举例:

  1. import java.util.ResourceBundle;
  2. public class IORead {
  3. public static void main(String[] args) {
  4. ResourceBundle resourceBundle = ResourceBundle.getBundle("db");
  5. String user = resourceBundle.getString("user");
  6. System.out.println(user);
  7. String password = resourceBundle.getString("password");
  8. System.out.println(password);
  9. }
  10. }

4. 总结:

  1. 原始的方式:写相对路径的话,无法跨编译器;因为不同的编译器默认相对的路径是不同的。写绝对路径的话,无法跨平台,因为不同操作系统的文件规则是不一样的,比如 Windows系统中的绝对路径是带盘符(D盘,C盘),Linux 系统中的文件规则是不带盘符的。当在J Windows 操作系统中编写的绝对路径的Java程序,移植到到 Linux 操作系统中就会报错。
  2. 静态获取的绝对路径 和 动态获取绝对路径。
  3. 上述的三种优化方式,都是动态获取绝对路径的,但是都是基于 类路径下的文件才行的,不同所读取的文件不在 类路径下 是无法动态获取到对应绝对路径的。
  4. 上述 :优化方式1,优化方式2 可以动态获取到 类路径下的任意文件信息。但是 优化方式三:只能获取到 类路径下的以 .properties 后缀的配置文件信息了。
  5. 注意:优化方式三:不可以写文件后缀名,直接写文件名就可以了。因为资源绑定器,就只能绑定 xxx.properties 配置文件 ,并且这个文件必须在 类路径下。
  6. 如果类路径下,比如:IDEA 中的 src 目录就是类路径,文件是直接在 src 类路径下没有包含子目录的话,可以直接写 文件名+文件后缀名,如果文件是在 src 目录下含有的子目录下,则需要指明 类路径 src 下的哪个子目录的文件。

5. 最后:

限于自身水平,其中存在的错误,希望大家给予指教,韩信点兵——多多益善 。谢谢大家,后会有期 ,江湖再见 !!!

Java 优化:读取配置文件 "万能方式" 跨平台,动态获取文件的绝对路径的更多相关文章

  1. java中读取配置文件ResourceBundle和Properties两种方式比较

    今天在开发的时候,需要把一些信息放到配置文件中,方便后续的修改,注意到用的是ResourceBundle读取配置文件的方式,记得之前也见过使用Properties的方式,就比较好奇这两种方式的区别,网 ...

  2. Java读取配置文件的方式

    Java读取配置文件的方式-笔记 1       取当前启动文件夹下的配置文件   一般来讲启动java程序的时候.在启动的文件夹下会有配置文件 classLoader.getResource(&qu ...

  3. python读取配置文件的方式

    python读取配置文件的方式 1.从config.ini中读取,后缀无所谓,文件名字也无所谓,不过config.ini是常用写法,所谓见名知意 config.ini内容: [global] ip = ...

  4. jstl删除session,choose,动态获取request当前工程路径

    1.jstl标签c:remove删除session request.getSession().setAttribute("ssmsg", "修改成功"); &l ...

  5. java如何从一段html代码中获取图片的src路径

    java如何从一段html代码中获取图片的src路径 package com.cellstrain.icell.Test; import java.util.ArrayList;import java ...

  6. fileReader对象读取txt文件乱码问题 以及如何获取文件的url路径(绝对路径)

    <input type="file" @change="aaa($event)"> <div id="hi">< ...

  7. delphi 动态获取文件类型的图标

    delphi 动态获取文件类型的图标.txt我不奢望什么,只希望你以后的女人一个不如一个.真怀念小时候啊,天热的时候我也可以像男人一样光膀子!在应用程序的编写中,组合框(ComboBox).列表框(L ...

  8. Java 加载配置文件的方式

    一 使用原生方式读取配置文件 1 文件系统加载 Java代码   InputStream in = new FileInputStream("config.properties") ...

  9. python中读取配置文件的方式

    方式1:argparse argparse,是Python标准库中推荐使用的编写命令行程序的工具.也可以用于读取配置文件. 字典样式的配置文件*.conf 配置文件test1.conf { " ...

  10. java中读取配置文件

    若是Javaweb项目,项目运行于tomcat或其他容器时,可以使用下面方式来获取文件的输入流 1.当属性文件放在src下面时 InputStream is = Thread.currentThrea ...

随机推荐

  1. Vue使用axios请求接口返回成功200但是进入到catch中

    发生这个问题时查阅了许多资料,没有一个是对得上的.最后发现原来是在请求拦截器中的错误 错误代码如下 // 添加响应拦截器 axios.interceptors.response.use(functio ...

  2. 【Flume】概述及组成、入门案例、进阶(事务、拓扑结构)、不同拓扑案例、自定义、数据流监控Ganglia

    一.概述 1.定义 日志采集.聚合.传输的系统,基于流式结构 即:读取本地磁盘数据,写入HDFS或kafka 2.架构 Agent:JVM进程,以事件形式将数据送到目的地. Agent由三部分组成:S ...

  3. 12V转5V降压芯片,12V转3.3V稳压芯片电路图

    12V转5V应用中,大多要求会输出电流高的,稳压LDO就不能满足了,需要使用DC-DC降压芯片来持续稳压5V,输出电流1000MA,2000MA,3000MA,5000MA等.不同的输出电流可以选择适 ...

  4. SQLMap入门——获取当前网站数据库的名称

    列出当前网站使用的数据库 python sqlmap.py -u http://localhost/sqli-labs-master/Less-1/?id=1 --current-db

  5. Python + logging 控制台有日志输出,但日志文件中数据为空

    源码: def output(self, level, message): fh = logging.FileHandler(self.logpath, mode='a', encoding='utf ...

  6. 数据结构 传统链表实现与Linux内核链表

    头文件: #pragma once #include<stdlib.h> //链表结点 struct LinkNode{ void *data; struct LinkNode *next ...

  7. Introduction & Directory

    一个日常划水的高中生而已啦,我不会承认这个博客的CSS是copy的 一个貌似并不准确的图-- <算法竞赛进阶指南>学习: 动态规划: DP经典例题--LIS&LCS 洛谷题解: 搜 ...

  8. [生命科学] 生物基础实验之DNA提取

    生物基础实验之DNA提取实验 基因组DNA的提取通常用于构建基因组文库.Southern杂交(包括RFLP)及PCR分离基因等.利用基因组DNA较长的特性,可以将其与细胞器或质粒等小分子DNA分离. ...

  9. 【随笔】Ubuntu18.04下virtualbox卡死的解决办法

    //得到该进程ID X pgrep Xorg //杀掉进程 kill X 然后重新登陆帐号即可

  10. 工作这么多年,我总结的数据传输对象 (DTO) 的最佳实践

    前言 数据传输对象 (DTO) 是一种设计模式,常用于软件开发不同层或者不同系统之间传输数据.DTO 的主要目的是封装数据并防止它被其他层或系统直接访问或修改.通过遵循一组最佳实践,开发人员可以确保他 ...