解析Java和IDEA中的文件打包问题

问题:想在IDEA中引用相对路径,但是找不到文件。

项目目录结构

当前项目的路径为:D:\source\java\test\

项目结构如下

当前路径

面对无法使用相对路径找到资源文件的问题,首先想到的解决办法是先瞄一眼IDEA在执行时给Java环境设定的当前路径在哪,也就是说看看我们在使用相对路径时到底是相对于哪里的。

应该咋写呢?下面是Java API中的一些描述。

默认情况下, java.io包中的类始终会根据当前用户目录解析相对路径名。 该目录由系统属性user.dir ,通常是调用Java虚拟机的目录。

根据上面的信息,可以想到两个办法来获得当前路径,第一种办法是向Java中传递空字符串,这是一个比较hack的方法,按照相对路径来解析的话,自然会被解析成当前目录。

File f1 = new File(""); // 空字符串,相当于当前路径
System.out.println(f1.getAbsolutePath());

第二种办法,直接获取系统属性中的user.dir

System.out.println(System.getProperty("user.dir"));

两种办法得到的结果一致:

也就是说,当前目录指向项目的根目录,如果你想要获取src中的一个文件的话,需要使用src/文件名,而如果该文件在某个包下的话,则需要使用src/完整包路径/文件名

现在我在src下创建一个文件text1.txt,写入内容HELLO , WORLD

File f2 = new File("src/text1.txt");
LineNumberReader reader2 = new LineNumberReader(new FileReader(f2));
System.out.println(reader2.readLine());

结果

这样写好吗?

我们的Java代码最终会被打包上传到目标平台,如某个服务器上,打包的代码应该只包含编译后的文件目录,在IDEA中就是那个out目录,其它的文件都不会出现在目标平台上,到那个时候,你不再拥有当前项目编写时的目录结构,你的程序运行时所在的当前路径也不再是现在这个。

简单来说,我们写出了依赖环境的代码,这个代码不保证程序运行在生产环境下的时候还可以正常执行,而且通常是无法正常执行。

唯一的办法,就是我们在IDEA编译时,告诉它有些文件,请你帮我打包到out目录中,然后我再想办法去out目录中找,这下就不会出问题了。

JavaAPI中有这样一个方法,它被明确说明是用来干这个事的就是ClassLoader中的getResource,下面是API中的说明

寻找给定名字的资源文件。一个资源文件可以是一些能够被class代码以一种独立于代码位置的方式进行访问的数据(图像、声音、文字等)

我们先看看使用这个方法是相对于哪个路径,采用的办法和new File("")大同小异。

String path = Main.class.getClassLoader().getResource("").getPath();
System.out.println(path);

结果如下

它已经找到了我们打包后的二进制代码所在的位置,这下就可以相对于二进制代码的位置读写文件了,不再依赖当前环境。

注意,ClassLoader.getResourceAsStream和Class.getResourceAsStream有些细微的区别,见JAVA 笔记 ClassLoader.getResourceAsStream() 与 Class.getResourceAsStream()的区别

将文件打包到二进制代码位置

创建文件夹,mark directory as -> Resources Root

o

在其中创建text2.txt,写入HELLO , RESOURCES

然后这样去读

LineNumberReader r3 = new LineNumberReader(
        new InputStreamReader(Main.class.getClassLoader().getResourceAsStream("text2.txt"))
);
System.out.println(r3.readLine());

这里使用了ClassLoader.getResource的一个变体getResourceAsStream,它的作用就是返回的是一个流,而非URL。我们所做的就是获取当前类的ClassLoader,通过它获取资源文件,转换成LineNumberReader。

读取成功,也就是说,在resources文件夹下的文件,IDEA会自动帮我们打包到二进制代码的位置

其他

src中的文件

如果我们去看out文件夹下的文件结构,你就会发现,除了resources/text2.txt,我们之前在src下创建的text1.txt也被打包到这个位置了。也就是说,我们可以通过同样的方式去读取src下的文件。

D:\source\java\test\out>wsl tree .
.
└── production
    └── test
        ├── io
        │   └── lilpig
        │       └── test
        │           └── Main.class
        ├── text1.txt
        └── text2.txt

5 directories, 3 files

LineNumberReader r4 = new LineNumberReader(
        new InputStreamReader(Main.class.getClassLoader().getResourceAsStream("text1.txt"))
);
System.out.println(r4.readLine());

读取成功

Thread.currentThread.getContextClassLoader

在多线程应用中,有可能你编写的类的类加载器和实际加载resources文件夹的类加载器不是同一个,这会导致你的代码无法获取到资源文件,使用Thread.currentThread.getContextClassLoader能规避这个问题。具体的原理涉及到Java中的类加载机制的委托模型,这部分的内容原理性太强,我已经忘得差不多了,就不露怯了。

我们编写的大部分后端应用,都是多线程应用,我们自己感知不到,那是因为多线程代码被封装在框架中,所以无论如何,使用Thread.currentThread.getContextClassLoader总比使用Main.class.getClassLoader更好。

并且,Thread.currentThread.getContextClassLoader更加具有一致性,你可以轻易的编写一个工具方法来简化代码(无论如何Thread.currentThread返回的总是当前调用者所在的线程)。而如果用之前的方法,每一个类的类名都不同,你在一个类中写的是Main.class.getClassLoader在另一个类中写的又是Other.class.getClassLoader,这不一致的代码会让我们难以编写一个通用的工具类。

框架

在框架中,资源文件夹都是默认被创建好的,有的可能叫static,有的可能叫resources......而且,有可能编译后的目录不是out,而是target,甚至会被打包成war包等。

我们要做的不是管它打包后在哪,而是直接往框架给你指定好的资源文件夹里面放文件就行,剩下的交给框架。

并且在框架中,随处可见Thread.currentThread.getClassLoader().getResource...这样的代码。

参考

Java SE 8 API

Thread.currentThread().getContextClassLoader() 和 Class.getClassLoader()区别

到此这篇关于Java和IDEA中的文件打包的文章就介绍到这了,更多相关Java和IDEA中的文件打包内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • 在idea中将java项目中的单个类打包成jar包操作

    JAR文件的全称是Java Archive File,即Java档案文件.JAR文件是一种压缩文件,与常见的ZIP压缩文件兼容,被称为JAR包. JAR文件与zip文件的主要区别是在JAR文件中默认包含了一个名为META-INF/MANIFEST.MF的清单文件,这个清单文件是在生成JAR文件时系统自动创建的. 打包jar包 1.先创建一个要打包成jar包的类 2.File -> Project Structrue -> Artifacts -> + -> JAR -> fr

  • JAVA 根据Url把多文件打包成ZIP下载实例

    压缩文件代码工具类: public class UrlFilesToZip { private static final Logger logger = LoggerFactory.getLogger(UrlFilesToZip.class); //根据文件链接把文件下载下来并且转成字节码 public byte[] getImageFromURL(String urlPath) { byte[] data = null; InputStream is = null; HttpURLConnec

  • javaweb文件打包批量下载代码

    本文实例为大家分享了javaweb文件打包批量下载,供大家参考,具体内容如下 // 批量下载未批改作业 @RequestMapping(value = "/downloadAllHomework", method = RequestMethod.GET) public void downloadAllHomework(HttpSession httpSession, HttpServletRequest request, HttpServletResponse response, St

  • IDEA打包jar-解决找不到或无法加载主类 main的问题

    学习大佬们开发安全小工具,打包jar解决错误: 找不到或无法加载主类 main 1 Maven方式 遇到报错"找不到或无法加载主类 main" 解决方案 一定加入<build> <plugins>中的插件,这里需要注意的是 <mainClass>Main</mainClass>,这里填写的路径为/src/main/java下开始写的 <?xml version="1.0" encoding="UTF-8

  • java实现批量下载 多文件打包成zip格式下载

    本文实例为大家分享了java实现批量下载的具体代码,供大家参考,具体内容如下 现在的需求的: 根据产品族.产品类型,下载该产品族.产品类型下面的pic包: pic包是zip压缩文件: t_product表: 这些包以blob形式存在另一张表中: t_imagefile表: 现在要做的是:将接入网.OLT下面的两个包downloadPIC:MA5800系列-pic.zip 和 MA5900-pic.rar一起打包成zip压缩文件下载下来: 代码: ProductController.java: /

  • java实现服务器文件打包zip并下载的示例(边打包边下载)

    使用该方法,可以即时打包文件,一边打包一边传输,不使用任何的缓存,让用户零等待! 复制代码 代码如下: /** *  * mySocket 客户端 Socket * @param file 待打包的文件夹或文件 * @param fileName 打包下载的文件名 * @throws IOException */ private void down(File file, String fileName) throws IOException { OutputStream outputStream

  • IDEA导出jar打包成exe应用程序的小结

    Java jar打包成exe应用程序,可在无JDK/JRE环境下运行 老师让做一个小项目,但是需要打包发布出来,因此在网上查了很多资料之后总结的经验. 1. IDEA导出jar包,选择 File -> Project Structure (快捷键:Ctrl+Alt+Shift+S). 2. 选中"Artifacts",点击"+"选择jar,然后选择"from modules with dependencies". 3. 选择文件图标,选中入

  • java文件下载代码实例(单文件下载和多文件打包下载)

    这篇文章主要介绍了java文件下载代码实例(单文件下载和多文件打包下载),文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 最近项目有需要写文件下载相关代码,这边提交记录下相关代码模块,写的不太好,后期再优化相关代码,有好的建议,可留言,谢谢. 1)单文件下载 public String oneFileDownload(HttpServletRequest request,HttpServletResponse response){ //针对需求需

  • 解析Java和IDEA中的文件打包问题

    问题:想在IDEA中引用相对路径,但是找不到文件. 项目目录结构 当前项目的路径为:D:\source\java\test\ 项目结构如下 当前路径 面对无法使用相对路径找到资源文件的问题,首先想到的解决办法是先瞄一眼IDEA在执行时给Java环境设定的当前路径在哪,也就是说看看我们在使用相对路径时到底是相对于哪里的. 应该咋写呢?下面是Java API中的一些描述. 默认情况下, java.io包中的类始终会根据当前用户目录解析相对路径名. 该目录由系统属性user.dir ,通常是调用Jav

  • java读取ftp中TXT文件的案例

    最近在开发关于java读取ftp中TXT文件,其中有些坑踩了一下,再次做个记录 1.读取文件时我会根据文件名称去生成数据库表,oracle数据库对于表名的长度是有限制的,最多30个字符 2.对于多个文件的ftp的读取,每次获取文件后再次回去文件的流会为空,即在循环中多次根据ftp获取文件的流 当出现这种情况时,需要在循环时每次开启和关闭ftp的链接即可解决,否则在第二次获取的时候inputsteam为null 3.读取txt文件时,如果文件中包含中文,进行读取时可能会出现乱码,这是可设置读取的字

  • Java如何实现将类文件打包为jar包

    目录 将类文件打包为jar包 1.写类文件 2.编译 3.测试 4.打jar包 jar包应该怎么打? 1.首先确保你的项目 2.选中你的项目,点右键 3.选择runnable jar file 4.如下图,直接看图 5.然后点finish 将类文件打包为jar包 为实际项目写了一个工具类,但是每次使用时都需要打开项目点击运行.突然想,不如将这个类打成jar包这样后续就可以直接运行了. 说做就做. 1.写类文件 大概就这么个样子. 注意,这里用到了外部依赖,也就是为什么要写这篇文章的原因. 本例中

  • 深入解析Java并发程序中线程的同步与线程锁的使用

    synchronized关键字 synchronized,我们谓之锁,主要用来给方法.代码块加锁.当某个方法或者代码块使用synchronized时,那么在同一时刻至多仅有有一个线程在执行该段代码.当有多个线程访问同一对象的加锁方法/代码块时,同一时间只有一个线程在执行,其余线程必须要等待当前线程执行完之后才能执行该代码段.但是,其余线程是可以访问该对象中的非加锁代码块的. synchronized主要包括两种方法:synchronized 方法.synchronized 块. synchron

  • 解析Java的迭代器中的fast-fail错误检测机制

    fail-fast 机制是java集合(Collection)中的一种错误机制.当多个线程对同一个集合的内容进行操作时,就可能会产生fail-fast事件.例如:当某一个线程A通过iterator去遍历某集合的过程中,若该集合的内容被其他线程所改变了:那么线程A访问集合时,就会抛出ConcurrentModificationException异常,产生fail-fast事件. fail-fast 机制是java集合(Collection)中的一种错误机制.当多个线程对同一个集合的内容进行操作时,

  • 解析Java和Eclipse中加载本地库(.dll文件)的详细说明

    最近在做的工作要用到本地方法,需要在Java中加载不少动态链接库(以下为方便延用Windows平台下的简写dll,但并不局限于Windows).刚刚把程序跑通,赶紧把一些心得写出来,mark.也希望对大家的类似工作有所帮助首先,应当明确,dll有两类:(1)Java所依赖的dll和,(2)dll所依赖的dll.正是由于第(2)种dll的存在,才导致了java中加载dll的复杂性大大增加,许多说法都是这样的,但我实验的结果却表明似乎没有那么复杂,后面会予以详细阐述.其次,Java中加载dll的方式

  • 深入解析Java设计模式编程中观察者模式的运用

    定义:定义对象间一种一对多的依赖关系,使得当每一个对象改变状态,则所有依赖于它的对象都会得到通知并自动更新. 类型:行为类模式 类图: 在软件系统中经常会有这样的需求:如果一个对象的状态发生改变,某些与它相关的对象也要随之做出相应的变化.比如,我们要设计一个右键菜单的功能,只要在软件的有效区域内点击鼠标右键,就会弹出一个菜单:再比如,我们要设计一个自动部署的功能,就像eclipse开发时,只要修改了文件,eclipse就会自动将修改的文件部署到服务器中.这两个功能有一个相似的地方,那就是一个对象

  • 解析Java线程编程中的线程安全与synchronized的使用

    一.什么时候会出现线程安全问题? 在单线程中不会出现线程安全问题,而在多线程编程中,有可能会出现同时访问同一个资源的情况,这种资源可以是各种类型的的资源:一个变量.一个对象.一个文件.一个数据库表等,而当多个线程同时访问同一个资源的时候,就会存在一个问题: 由于每个线程执行的过程是不可控的,所以很可能导致最终的结果与实际上的愿望相违背或者直接导致程序出错. 举个简单的例子: 现在有两个线程分别从网络上读取数据,然后插入一张数据库表中,要求不能插入重复的数据. 那么必然在插入数据的过程中存在两个操

随机推荐