Java实现动态获取文件的绝对路径
作者:Rainbow-Sea 发布时间:2022-12-19 00:34:14
前言
我们知道在 Java 中读取一些配置文件信息,是在开发中十分常用的要求。
例如:这里我们使用 JDBC 实例:连接MySQL 数据库,读取连接数据库的 用户名,密码 。
如下是一个名为 jdbc.properties
的配置文件信息,以及存在目录
package blogs.blogs8;
import java.io.FileInputStream;
import java.io.IOException;
import java.util.Properties;
public class IORead {
public static void main(String[] args) {
FileInputStream f = null;
try {
// 创建字节输入流对象
// 在IDEA 中的默认相对路径是在 src 同级目录下的
f = new FileInputStream("src/blogs/blogs8/jdbc.properties");
// 创建Map集合中的 Properties 对象
Properties properties = new Properties();
properties.load(f);
// 通过 key 读取对应的键值对
String user = properties.getProperty("user");
System.out.println(user);
String password = properties.getProperty("password");
System.out.println(password);
} catch (IOException e) {
e.printStackTrace();
} finally {
// 关闭IO资源
if(f == null) {
try {
f.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
说明:
上述的读取文件的方式,我们可以看到是 “完全没有问题的” 可以读取到对应的配置信息,但是存在一个缺点:就是移除性差。src 中是在 IDEA 这个编译器中体现的,如果是在其它的编译器中运行的时候,很大的可能会报错,原因是:这里我们使用的相对路径是,在 IDEA中的,IDEA 中的默认相对路径是 在 project 下的也就是 src 的同级目录。但是其它的系统,或者编译器就可能不是这个和 IDEA 中默认相对路径了。运行程序时,就有可能会报错:如下:找不到指定的文件。
上述这种方式:如果我们不写相对路径,而是写绝对路径的话,也是存在一个问题的。那就是因为该绝对路径是写死了的,不是动态获取的,该路径在 Windows 操作系统中是存在盘符的,所以写绝对路径的时候是需要带上盘符(E盘,D盘的),但是如果该程序是运行在其他操作系统中的话,比如 Linux 操作系统中是没有盘符的说法的。所以就会出问题。无法跨平台。
1. 优化方式一:返回一个文件的绝对路径
接下来说一种比较通用的一种路径:即使代码换位置了,这样的代码编写的方式仍然是通用的。因为该文件的路径是动态获取的。
在Windows中的话,就以该系统的文件规则,动态获取到的绝对路径是带盘符的,而 Linux系统中就以该系统的文件规则,获取到的绝对路径是不带盘符的。 这就可以跨平台了。
注意: 使用该方式的前提是:所读取的文件必须是在 类路径
下才行。如果不是在类路径下,运行程序时是会报错:系统找不到指定的路径
。
什么是类路径 ?
类路径也是一种特殊的相对路径,只不过它相对的是class文件。在 IDEA 中的类路径是在 src 目录下的。重点记住它
该方式的核心代码:
String path = Thread.currentThread().getContextClassLoader().getResource("db.properties").getPath();
/*
解释:
Thread.currentThread() 当前线程对象
getContextClassLoader() 是线程对象的方法,可以获取到当前线程的类加载对象
getResource() 获取资源:这是类加载器对象的方法,当前线程的类加载器默认从类的根路径下加载资源。
getPath() 获取当文件的绝对路径
*/
1.1 情况一
所读取的文件是直接存放在 src 的目录下的,该文件的并没有其它的的包。如下图所示:可以直接写文件名 + 文件名的后缀即可。
import java.io.FileInputStream;
import java.io.IOException;
import java.util.Properties;
public class IORead {
public static void main(String[] args) {
String path = Thread.currentThread().getContextClassLoader().getResource("db.properties").getPath();
System.out.println(path); // 返回该文件的绝对路径:
}
}
通过该方式获取到指定文件的绝对路径,再将该绝对路径,作为参数,创建FileInputStream
字节输入流对象
import java.io.FileInputStream;
import java.io.IOException;
import java.util.Properties;
public class IORead {
public static void main(String[] args) {
FileInputStream f = null;
try {
// 获取到该配置文件的的绝对路径
String path = Thread.currentThread().getContextClassLoader().getResource("db.properties").getPath();
// 通过该获取的文件的绝对路径创建 字节输入流对象
f = new FileInputStream(path);
// 创建Map集合中的 Properties 对象
Properties properties = new Properties();
properties.load(f);
// 通过 key 读取对应的键值对
String user = properties.getProperty("user");
System.out.println(user);
String password = properties.getProperty("password");
System.out.println(password);
} catch (IOException e) {
e.printStackTrace();
} finally {
// 关闭IO资源
if (f != null) {
try {
f.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
也是可以读取到文件中是在 src 目录下。
1.2 情况二
当所读取的文件,是在 src 目录下,但是该 src 目录下还有其他的包(目录),则不可以直接写 “文件名+ 文件后缀名”了,而是需要写明该 src 包(目录)下的 相对路径:如下图所示的文件:该路径名应该是:blogs/blogs8/jdbc.properties
举例:
import java.io.FileInputStream;
import java.io.IOException;
import java.util.Properties;
public class IORead {
public static void main(String[] args) {
FileInputStream f = null;
try {
// 获取到该配置文件的的绝对路径,如下src目录下还有目录(包),需要指定 src目录下/包下的哪个文件。
String path = Thread.currentThread().getContextClassLoader().getResource("blogs/blogs8/jdbc.properties").getPath();
// 通过该获取的文件的绝对路径创建 字节输入流对象
f = new FileInputStream(path);
// 创建Map集合中的 Properties 对象
Properties properties = new Properties();
properties.load(f);
// 通过 key 读取对应的键值对
String user = properties.getProperty("user");
System.out.println(user);
String password = properties.getProperty("password");
System.out.println(password);
} catch (IOException e) {
e.printStackTrace();
} finally {
// 关闭IO资源
if (f != null) {
try {
f.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
2. 优化方式二:返回一个 InputStream 字节输入流
上述方式一:我们需要通过 :new 一个 FileInputStream 字节输入流对象的方式,这里我们直接通过指定的文件名的,直接返回一个 InputStream 字节输入流 ,不需要 new 。
同样的:该读取的文件必须是在类路径下才行,这里的IDEA的类路径是 src 目录下
核心代码如下:
// 直接以流的形式返回。
InputStream resourceAsStream = Thread.currentThread().getContextClassLoader().
getResourceAsStream("db.properties");
举例:
import java.io.IOException;
import java.io.InputStream;
import java.util.Properties;
public class IORead {
public static void main(String[] args) {
// 直接在 src目录下没有包含任何子目录,可以直接写文件名+ 后缀,而如果有子目录,需要指明子目录下的文件名+后缀名
InputStream inputStream = Thread.currentThread().getContextClassLoader().getResourceAsStream("db.properties");
// 创建 Properties 集合对象,通过流获取指定配置文件中的键值对信息
Properties properties = new Properties();
try {
properties.load(inputStream);
} catch (IOException e) {
e.printStackTrace();
}
String user = properties.getProperty("user");
System.out.println(user);
String password = properties.getProperty("password");
System.out.println(password);
// 关闭IO资源
if (inputStream != null) {
try {
inputStream.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
3. 优化方式三:java.util 包下提供了一个资源绑定器
上述两个方式可以获取到任意文件的信息。
但是以下这个方式三:就只能获取到 类路径下的以 .properties
后缀的配置文件信息了。
java.util
包下提供了一个资源绑定器,便于获取属性.properties
配置文件中的内容。
该资源绑定器:只能绑定 xxx.properties
配置文件 ,并且这个文件必须在 类路径下,这里的 IDEA 是 src 目录下。
并且在写路径的时候,路径后面的扩展名不能写,写了会报错: ``。因为既然只能读取 properteis 后缀的文件,那就不用再多余的写文件后缀名了。
如果在 src 目录下的子目录中的文件,需要指明是 src 下的哪个子目录下的文件,同样不要写文件后缀名,不然报错。
举例:
import java.util.ResourceBundle;
public class IORead {
public static void main(String[] args) {
ResourceBundle resourceBundle = ResourceBundle.getBundle("db");
String user = resourceBundle.getString("user");
System.out.println(user);
String password = resourceBundle.getString("password");
System.out.println(password);
}
}
4. 总结
原始的方式:写相对路径的话,无法跨编译器;因为不同的编译器默认相对的路径是不同的。写绝对路径的话,无法跨平台,因为不同操作系统的文件规则是不一样的,比如 Windows系统中的绝对路径是带盘符(D盘,C盘),Linux 系统中的文件规则是不带盘符的。当在J Windows 操作系统中编写的绝对路径的Java程序,移植到到 Linux 操作系统中就会报错。
静态获取的绝对路径 和 动态获取绝对路径。
上述的三种优化方式,都是动态获取绝对路径的,但是都是基于 类路径下的文件才行的,不同所读取的文件不在 类路径下 是无法动态获取到对应绝对路径的。
上述 :优化方式1,优化方式2 可以动态获取到 类路径下的任意文件信息。但是 优化方式三:只能获取到 类路径下的以
.properties
后缀的配置文件信息了。注意:优化方式三:不可以写文件后缀名,直接写文件名就可以了。因为资源绑定器,就只能绑定
xxx.properties
配置文件 ,并且这个文件必须在 类路径下。如果类路径下,比如:IDEA 中的 src 目录就是类路径,文件是直接在 src 类路径下没有包含子目录的话,可以直接写
文件名+文件后缀名
,如果文件是在 src 目录下含有的子目录下,则需要指明 类路径 src 下的哪个子目录的文件。
来源:https://www.cnblogs.com/TheMagicalRainbowSea/p/17143604.html


猜你喜欢
- 下面介绍的这个版本搭配是我研究好久好久才跑通的,这在我的电脑上是一组可行的配置,如果你使用了同样的配置跑不通,那可能是环境中某一部分还是有不
- 有时我们可能会遇到下图这样一种情况 — 我们需要的资料或教程被分成了几部分存放在多个PDF文件中,不管是阅读还是保存都不是很方便,这时我们肯
- 在android中的webview中,可以对文本内容进行对齐,具体方法如下 public class MainActivity
- Date类概述java.util.Date类 表示特定的瞬间,精确到毫秒。 继续查阅Date类的描述,发现Date拥有多个构造函数,只是部分
- 本文记录一下,我从AndroidStudio 2.3.3升级到3.0,再升级到3.0.1一路上遇到的输入法之坑以及解决方案。前些天把Andr
- 平时,我们将c#中的Distinct大多用于对数组去重,一般数组为基础的数据类型,例如 int,string.也可以用于对象去重,我们看看C
- 要想了解Java * ,首先要了解什么叫做代理,熟悉设计模式的朋友一定知道在Gof总结的23种设计模式中,有一种叫做代理(Proxy)的对
- 这篇文章主要讲述服务追踪组件zipkin,Spring Cloud Sleuth集成了zipkin组件。一、简介Add sleuth to
- 银行卡大家都使用,在密码输错超过限制次数之后,就容易被锁死,智能到银行柜台才能解锁,那么这一功能如果实现的呢,今天小编通过实例代码给大家详细
- 一、概述 在C和C++语言中都有assert关键,表示断言。在Java中,同样也有assert关键字,表示断言,用法和含义都差不多
- Socket异常客户端异常java.net.ConnectException: Connection refused: connect。该异
- java -version 命令大家都用过,大部分就是看下jdk版本或检查下环境变量的设置,但最后一行的信息也挺重要,如下图所示:Serve
- final关键字可用于变量声明,一旦该变量被设定,就不可以再改变该变量的值。 通常final定义的变量为常量。如:final double
- 本文向大家分享的主要内容是Java面试中一个常见的知识点:volatile关键字。本文详细介绍了volatile关键字的方方面面,希望大家在
- 本文实例为大家分享了Android本地实现搜索历史记录的具体代码,供大家参考,具体内容如下一.自定义搜索历史记录本地实现搜索历史记录有很多种
- 实例如下:/// <summary> /// 上传ftp服务 /// </summary>
- 目录一、Shiro简介 核心角色核心理念二、整合SpringBoot2框架 1、核心依赖2、Shiro核心配置3、域对象
- 今天带大家实现滑动返回效果.,具体内容如下所示:先看看效果图:因为没有具体内容,也没有简书的图片资源,所以稍微简陋了点.但是依然不妨碍我们的
- 本文实例讲述了Android弹出窗口实现方法。分享给大家供大家参考,具体如下:直接上代码:/*** 弹窗--新手指引* @param cxt
- 微信开放平台 : https://open.weixin.qq.com/一、准备工作 : 1. Android Studio环境下:在bui