classloader.getresources() 介绍

◆普通情况下,我们都使用相对路径来获取资源,这种灵活性比較大.

比方当前类为com/bbebfe/Test.class

而图像资源比方sample.gif应该放置在com/bbebfe/sample.gif

而假设这些图像资源放置在icons文件夹下,则应该是com/bbebfe/icons/sample.gif

通过当前类文件的路径获取资源主要有例如以下几种方式:

· 如果当前类为com.bbebfe.Test

· 包所在的目录为bin

String
imageName
 =
"icons/sample.gif"

1, 通过Class.getResource()定位类路径下的资源(bin/com/bbebfe/icons/sample.gif)

Class clazz = this.getClass();

URL url = clazz.getResource(imageName);

2,通过ClassLoader.getResource()定位包的根文件夹下的资源(bin/icons/sample.gif)

Class clazz =
this.getClass();

URLClassLoader
loader = (URLClassLoader)clazz.getClassLoader();

URL url =
loader.getResource(imageName);

3, 通过ClassLoader.findResource()提供自己定制的方式定位资源

URL url = loader.findResource(imageName);

◆那么这三种方法有那些差别, 我们应该在何时使用哪种方法呢?

· Class.getResource() 方法

该方法实际通过该Class的Class
Loader的getResource()方法来获得资源, 在调用ClassLoader的getResource()方法之前, Class.getResource()方法会对资源名称做一定的处理,构建一个该资源的绝对名称(absolute name, 大意是:

+假设资源名称以‘/‘(‘"u002f‘) 開始, 则资源的绝对名称是‘/‘以后的部分.

假设imageName是"/icons/sample.gif", 则在这里会变成"icons/sample.gif"

+否则对于其它情况, 绝对名称将是例如以下形式(给资源名称的前面加上modified_package_name/):
             
modified_package_name/resource_name (修正的包名称/资源名称)

当中修正的包名称含义是将当前对象所在的包名称中的‘.‘(‘"u002e‘)替换为‘/‘

假设ClassLoader.getResource()方法返回一个值为null的URL, 则Class.getResource()方法终于会将资源请求交给ClassLoader.getSystemResource(java.lang.String).

· ClassLoader.getResource() 方法

该对资源进行查找, 资源的名称是以‘/‘分隔的路径, 这种方法首先查找自己的父亲ClassLoader,由自己的父ClassLoader来查找资源(实际上, 假设父亲的父亲不是空, 则父亲仍会向上提交查找请求). 假设自己的父ClassLoader是null, 则查找Java虚拟机中内建的class loader, 并将资源请求提交给它们, 假设这些操作都失败了, 则ClassLoader会调用自己的findResource()方法来查找资源.

· ClassLoader.findResource() 方法

该方法在内部查找指定的资源, 假设你实现了自己的Class Loader,则应该重载这种方法以自己特定的方式来查找类文件和资源.

◆通过以上的总结, 我们能够看到三点.

1, 不管是getResource(), 还是findResource(), 这些方法都仅仅是资源的定位方法, 终于都仅仅是返回一个URL, 仅仅是对资源的定位而已, 我们随后应通过自己的方法来读取这些资源. 而在Class和ClassLoader中还定义的有getResourceAsStream方法, 该方法是getResource的增强版, 这里就不介绍了.

2,假设须要以类为相对路径查找资源, 则应该调用Class.getResource()方法, 不要直接调用ClassLoader.getResource()方法. 另外, 除非是你自定义了ClassLoader并重载了findResource方法,否则也不要直接调用ClassLoader.findResource方法, 由于在Class.getResource()方法中会对资源名称作一定的处理, 这在上面介绍了, 以下举个实例:

如果我的当前类在EclipseprojectDatabase下, 类所在的包是com.bbebfe.test, 而icons文件夹放在bin/com/bbebfe/test/文件夹下, 我须要得到icons/sample.gif文件的URL, 则调用this.getClass().getResource()得到的URL是:

file:/E:/MyLife/MyProjects/Eclipse3.2/Database/bin/com/bbebfe/test/icons/disremove.gif

 

3, 有时候我们希望某个jar库的图像资源在同一个icons下统一管理, 而不是为每一个包以下的Class建一个icons, 也就是说须要以库为相对路径来查找资源, 此时则应该调用ClassLoader.getResource()方法, 举个样例:

·某个project有例如以下的包结构:

com.bbebfe.ui

com.bbebfe.test

com.bbebfe.database

·假设以类为相对路径, 则在每一个包下都必须建立一个icons文件夹, 并放置对应的资源文件. 例如以下:

com.bbebfe.ui/icons/...

com.bbebfe.test/icons/...

com.bbebfe.database/icons/...

·而我们可能希望在根文件夹下放置一个icons文件夹, 把全部资源放置在这里管理, 这样还能够防止资源的反复. 就是例如以下形式

com.bbebfe.ui

com.bbebfe.test

com.bbebfe.database

icons/sample.gif ...

则此时我们应该调用ClassLoader.getResource方法, 因为它没有对资源名称作处理, 也就是说没有将修正的包名加入到资源名称前, 所以它会在类所在的包的根下去查找资源.(执行java程序的语法是java
com.bbebfe.ui.Test, 所以根文件夹是com文件夹的上级文件夹).

◆最后, 在Java中对资源进行定位的方法有非常多种, 在Eclipse源码中还有例如以下一段定位文件资源的代码, 还没有时间研究, 以后再谈:

ProtectionDomain domain = Main.class.getProtectionDomain();

CodeSource source = null;

URL result = null;

if (domain != null)

source = domain.getCodeSource();//获得code source

if (source != null)

result = source.getLocation();//获得URL

String path = decode(result.getFile());//

// normalize to not have leading / so we can
check the form

File file = new File(path);

path = file.toString().replace(‘""‘,
‘/‘);

// create a file URL (via File) to normalize the
form (e.g., put

// the leading / on if necessary)

path = new File(path).toURL().getFile();

刚才试了一下,发现假设类路径上有反复的资源,getResource()方法会返回类路径上碰到的第一个资源。

而getResources()则会返回当前类载入器路径上的全部反复资源以及父类载入器上的全部反复资源。

比方,在tomcat/lib文件夹下放置一个zip文件,包括config/aaa.txt文件, 
在WEB-INF/lib文件夹下复制一份这个zip文件, 
再在src文件夹下放上config/aaa.txt文件,

getResource("config/aaa.txt")返回结果是: 
file:/D:/eclipse/workspace/demo/WebContent/WEB-INF/classes/config/aaa.txt

getResources("config/aaa.txt")返回结果是: 
file:/D:/eclipse/workspace/demo/WebContent/WEB-INF/classes/config/aaa.txt 
jar:file:/D:/apache-tomcat-6.0.16/lib/aaaaa.jar!/config/aaa.txt 
jar:file:/D:/eclipse/workspace/demo/WebContent/WEB-INF/lib/bbbbb.jar!/config/aaa.txt

时间: 2024-12-31 03:51:03

classloader.getresources() 介绍的相关文章

ClassLoader简单介绍

要理解ClassLoader,我们可以通过what.how两个方面来解释 一.what:什么事ClassLoader? 1.ClassLoader可以是将class文件加载到JVM方法区. 2.ClassLoader主要分为2类,用户自定义的类加载器和内部类加载器:启动内装载器(bootstrap)和用户自定义装载器(user-defined class loader). 启动类装器分为3类:Bootstrap ClassLoader:启动类加载器,是Java类加载层次中最顶层的类加载器,是Ex

Class.getResources()和classLoader.getResources()区别

Class.getResource(String path) path不以'/'开头时,默认是从此类所在的包下取资源: path 以'/'开头时,则是从ClassPath根下获取: package testpackage; public class TestMain { public static void main(String[] args) { System.out.println(TestMain.class.getResource("")); System.out.print

classloader原理

1.classLoader的介绍及加载过程 与普通程序不同的是,Java程序(class文件)并不是本地的可执行程序.当运行Java程序时,首先运行JVM(Java虚拟机),然后再把Java class加载到JVM里头运行,负责加载Java class的这部分就叫做Class Loader.所以classLoader的目的在于把class文件装入到jvm中. 那么classLoader又在那里的啦?又由谁调用呢?其实classLoader只是jvm的一个实现的一部分.Jvm提供的一个顶级的cla

(转)Java ClassLoader详解

转:http://java.chinaitlab.com/base/804400.html 类加载器是 Java 语言的一个创新,也是 Java 语言流行的重要原因之一.它使得 Java 类可以被动态加载到 Java 虚拟机中并执行.类加载器从 JDK 1.0 就出现了,最初是为了满足 Java Applet 的需要而开发出来的.Java Applet 需要从远程下载 Java 类文件到浏览器中并执行.现在类加载器在 Web 容器和 OSGi 中得到了广泛的使用.一般来说,Java 应用的开发人

classloader 学习

classloader就是把类文件加载到jvm中供虚拟机使用,先看一个magic小例子: 首先,我定义一个alex/vicky包,然后在这个包内定义一个接口: public interfaceIService { voidservice(); } 然后使用这个接口定义一个实现类: public classService extendsIService{ @Override publicvoidservice(){System.out.println("Alex"); } } 然后把这个

android classloader双亲托付模式

概述 ClassLoader的双亲托付模式:classloader 按级别分为三个级别:最上级 : bootstrap classLoader(根类载入器) : 中间级:extension classLoader (扩展类载入器) 最低级 app classLoader(应用类载入器). 根(Bootstrap)类载入器:该载入器没有父载入器.它负责载入虚拟机的核心类库,如java.lang.*等.比如java.lang.Object就是由根类载入器载入的.根类载入器从系统属性sun.boot.

深入浅出 JVM ClassLoader

# 前言 在 JVM 综述里面,我们说,JVM 做了三件事情,Java 程序的内存管理, Java Class 二进制字节流的加载(ClassLoader),Java 程序的执行(执行引擎).我们也说,我们大部分情况下只关注前2个.在前面的文章中,我们已经分析了内存关系相关的,包括运行时数据区,GC 相关.今天我们要讲的就是类加载器. 在 JVM 综述 里,我们已经大致分析了一些概念.而今天的文章将详细的阐述类加载器. 首先,我们要了解类加载器,当然,了解的目的是为了更好的开发,通过对类加载器的

深入Spring Boot:ClassLoader的继承关系和影响

前言 对spring boot本身启动原理的分析, Spring boot里的ClassLoader继承关系 可以运行下面提供的demo,分别在不同的场景下运行,可以知道不同场景下的Spring boot应用的ClassLoader继承关系. 分三种情况: 在IDE里,直接run main函数 则Spring的ClassLoader直接是SystemClassLoader.ClassLoader的urls包含全部的jar和自己的target/classes ========= Spring Bo

Android热修复原理普及

Android热修复原理普及 这段时间比较难闲,就抽空研究一下Android热修复的原理.自从Android热修复这项技术出现之后,随之而现的是多种热修复方案的出现.前两天又看到一篇文章分析了几种热修复方案的比较. 原文地址是:[Android热修复] 技术方案的选型与验证 看完这篇文章,有点汗颜.有这么多的热修复方案,并且他们之间的实现原理也不一样,各有优缺点. 然后在尼古拉斯_赵四的博客中看到几篇关于热修复的文章,对着这几篇文章撸了一番.大概的了解了热修复一种原理,其思路和QQ空间提出的安卓