ClassLoader.getSystemResourceAsStream("a.txt")获取不到资源文件

一、解决方案

换成XXX.class.getClassLoader().getResourceAsStream("a.txt")即可。

二、场景复现

src/main/resource下存在文件a.txt,项目类中static 方法中读取该文件。ClassLoader.getSystemResourceAsStream("a.txt")获取不到资源文件

三、原因剖析

getResourceAsStream会先使用本类的类加载器去加载,本类没有类加载器,才会使用系统类加载器。也就是说getResourceAsStream功能覆盖了getSystemResourceAsStream,所以推荐直接使用getResourceAsStream就完事了。都不需要知道具体两者有啥区别。

下面我们来细看有啥区别....

3.1 java类加载器

  • 1.启动类加载器(Bootstrap ClassLoader):顶层的类加载器,没有父类加载器。负责加载 /lib 目录下的,或被 -Xbootclasspath 参数所指定路径中的,并被 JVM 识别的(仅按文件名识别,如 rt.jar,名字不符合的类库即使放在 lib 目录也不会被加载)类库加载到虚拟机内存中。所有被 Bootstrap classloader 加载的类,它的 Class.getClassLoader 方法返回的都是 null,所以也称作 NULL ClassLoader。
  • 2.扩展类加载器(Extension CLassLoader):由 sun.misc.Launcher$ExtClassLoader 实现,负责加载 <JAVA_HOME>/lib/ext 目录下,或被 java.ext.dirs 系统变量所指定的目录下的所有类库;
  • 3.应用程序类加载器(Application/System ClassLoader):由 sun.misc.Launcher$AppClassLoader 实现。它是 ClassLoader.getSystemClassLoader() 方法的默认返回值,所以也称为系统类加载器(System ClassLoader)。它负责加载 classpath 下所指定的类库,如果应用程序没有自定义过自己的类加载器,一般情况下这个就是程序中默认的类加载器。

3.2 tomcat容器下类加载器

Tomcat官方说明:Class Loader HOW-TO

跟普通的java程序相比,类加载大体顺序相同。

  • 1.Bootstrap 启动类加载器: 加载JVM启动所需的类+系统扩展目录($JAVA_HOME/jre/lib/ext)里 JAR 文件中的类。
  • 2.System 系统类加载器:从 CLASSPATH 系统变量指定的目录中加载类库。该加载器加载的类对 tomcat 本身和 web 应用都可见。但是,标准的 tomcat 启动脚本($CATALINA_HOME/bin/catalina.sh or %CATALINA_HOME%\bin\catalina.bat)都会忽略系统变量 CLASSPATH 的值,而会使用如下的类库来创建 System 类加载器:

      $CATALINA_HOME/bin/bootstrap.jar

    $CATALINA_BASE/bin/tomcat-juli.jar 或 $CATALINA_HOME/bin/tomcat-juli.jar

    $CATALINA_HOME/bin/commons-daemon.jar

  • 3.Common 通用类加载器:通过该类加载器加载的类库可被 Tomcat 和所有应用共享。该类加载器的搜索位置是通过 $CATALINA_BASE/conf/catalina.properties 文件中的 common.loader 属性指定的,默认包括如下位置:

    $CATALINA_BASE/lib 下未打包的类和资源;

     $CATALINA_BASE/lib 下的 jar 包;

    $CATALINA_HOME/lib 下未打包的类和资源;

    $CATALINA_HOME/lib 下的 jar 包。

  • 4.WebappX 应用类加载器:每个 Web 应用创建一个自己的类加载器,加载自己项目下的数据: /WEB-INF/classes 和 /WEB-INF/lib 下的类和资源。并且不使用双亲委派机制,先自己加载,加载不到才使用父类加载器。

本来顺序是1234,但是WebappX不使用委派机制而是先自己加载,加载不了才使用父类,所以真实的顺序是:

  • Bootstrap classes of your JVM
  • /WEB-INF/classes of your web application
  • /WEB-INF/lib/*.jar of your web application
  • System class loader classes (described above)
  • Common class loader classes (described above)

tomcat8支持委托:配置允许委派:<Loader delegate="true"/>,顺序变为:

  • Bootstrap classes of your JVM
  • System class loader classes (described above)
  • Common class loader classes (described above)
  • /WEB-INF/classes of your web application
  • /WEB-INF/lib/*.jar of your web application

3.3 问题回归

tomcat容器中运行的java程序,使用系统类加载器是不能获取到资源的,必须使用WebappClassLoader。使用getResourceAsStream获取当前类的类加载器,也就是WebappClassLoader,自然可以获取到资源了。


=======参考=========

https://blog.csdn.net/w1196726224/article/details/54428493

原文地址:https://www.cnblogs.com/dennyzhangdd/p/11387200.html

时间: 2024-10-10 11:03:04

ClassLoader.getSystemResourceAsStream("a.txt")获取不到资源文件的相关文章

ClassLoader.getSystemResourceAsStream()

一: 要加载的文件和.class文件在同一目录下,例如:com.x.y 下有类Test.class ,同时有资源文件config.properties 那么,应该有如下代码: //前面没有"/"代表当前类的目录 InputStream is1 = Test.class.getResourceAsStream("config.properties"); System.out.println(is1);// 不为null   第二:在Test.class目录的子目录下,

className.class.getResourceAsStream与ClassLoader.getSystemResourceAsStream区别

className.class.getResourceAsStream : 一: 要加载的文件和.class文件在同一目录下,例如:com.x.y 下有类Test.class ,同时有资源文件config.properties 那么,应该有如下代码: //前面没有"/"代表当前类的目录 InputStream is1 = Test.class.getResourceAsStream("config.properties");System.out.println(is

web页面找不到资源文件,报404,但是资源文件存在且路径没错

如题 , 今天遇到这个问题,maven项目导入本地myeclipse,正常跑起来之后,在web端存在部分页面资源加载不进来. 但是项目资源确实存在,一开始以为是myeclipse开发环境搭建错误导致,一番整改之后还是报错,于是开始怀疑是开发工具的问题 换了一个eclipse之后还是存在该问题. 于是遇到下面的文章 web页面找不到资源文件,报404,但是资源文件存在且路径没错 经过一番修改 之后 问题解决 这个问题  其实没有我想的那么复杂 就是一个逻辑思维  陷入了僵局  既然资源加载不进来 

在DELPHI中*.wav 文件怎么加到资源文件中

比较“流行”的说法是:“16位的Delphi   1.0和32位的Delphi2.0.3.0都提供了资源         编译工具,其中   Delphi   1.0的资源编译器叫BRCC.EXE,Delphi   2.0的资源编译器         叫BRCC32.EXE   用来编译32位资源,所有资源编译器都只提供了命令行版本,         没有提供Windows   版本.制作一个资源一般要通过以下几个步骤:         1)编写.RC文件         ..RC文件是资源的源

Linux 获取设备树源文件(DTS)里描述的资源【转】

转自:http://www.linuxidc.com/Linux/2013-07/86839.htm 转自:http://blog.sina.com.cn/s/blog_636a55070101mced.html 在linux使用platform_driver_register() 注册  platform_driver 时, 需要在 platform_driver 的probe() 里面知道设备的中断号, 内存地址等资源. 这些资源的描述信息存放在 resource 数据结构中, 相同的资源存

C#后台代码获取程序集资源文件

资源会被打包在程序集内部. 选择这种生成方式后,该资源文件会被嵌入到该应用的程序集中,就是说打开生成的应用程序目录是看不到这个文件的. 可以用相对于当前的XAML文件的相对Uri访问,<Image Source="sl.png" />或是<Image Source="./sl.png" />, 在子文件夹里的可以用<Image Source="./images/sl.png" />访问到. 最保险的方式是采用特

cocos2d-x 找不到资源文件问题

问题描述: 在项目中引用到了图片,但是运行时报错 Possible missing file. 代码: Sprite _player = Sprite::create("player-hd.png"); 解决方法: 右键点击项目的属性,"VC++ Directories">"Source Directories"中增加"(ProjectDir)../Resources"

asp.net 多语言 在IIS7.5发布出现找不到资源文件

我也遇到这个问题,纠结了半天, 最后把资源文件的属性改为:内容 就可以了. 见:http://q.cnblogs.com/q/60443/

【已解决】IIS搭建 asp.net core 项目后 其他电脑访问不到资源文件

IIS搭建asp.net core 项目后,访问不到里面的资源文件(图片等),解决方法如下: 1.检查asp.net core发布文件中的资源文件是不是都放到了wwwroot名称的目录中. 2.检查asp.net core项目内Startup.cs文件中加上了app.UseStaticFiles(); 检查以上两点后 然后访问就可以找到资源了. ps 访问资源的链接中不用加 wwwroot. 原文地址:https://www.cnblogs.com/YangPeng/p/8971374.html