`
blueram
  • 浏览: 763613 次
  • 性别: Icon_minigender_1
  • 来自: 郑州
社区版块
存档分类
最新评论

Spring加载resource时classpath*:与classpath:的区别

 
阅读更多

Spring可以通过指定classpath*:与classpath:前缀加路径的方式从classpath加载文件,如bean的定义文件.classpath*:的出现是为了从多个jar文件中加载相同的文件.classpath:只能加载找到的第一个文件.

比如 resource1.jar中的package 'com.test.rs' 有一个 'jarAppcontext.xml' 文件,内容如下:

<bean name="ProcessorImplA" class="com.test.spring.di.ProcessorImplA" />

resource2.jar中的package 'com.test.rs' 也有一个 'jarAppcontext.xml' 文件,内容如下:

<bean id="ProcessorImplB" class="com.test.spring.di.ProcessorImplB" />

通过使用下面的代码则可以将两个jar包中的文件都加载进来

ApplicationContext ctx = new ClassPathXmlApplicationContext( "classpath*:com/test/rs/jarAppcontext.xml");

而如果写成下面的代码,就只能找到其中的一个xml文件(顺序取决于jar包的加载顺序)

ApplicationContext ctx = new ClassPathXmlApplicationContext( "classpath:com/test/rs/jarAppcontext.xml");

classpath*:的使用是为了多个component(最终发布成不同的jar包)并行开发,各自的bean定义文件按照一定的规则:package+filename,而使用这些component的调用者可以把这些文件都加载进来.

classpath*:的加载使用了classloader的 getResources() 方法,如果是在不同的J2EE服务器上运行,由于应用服务器提供自己的classloader实现,它们在处理jar文件时的行为也许会有所不同。 要测试classpath*: 是否有效,可以用classloader从classpath中的jar文件里加载文件来进行测试:getClass().getClassLoader().getResources("<someFileInsideTheJar>")。(上面的例子是在sun的jre中运行的状态)

 从Spring的源码,在PathMatchingResourcePatternResolver类中,我们可以更清楚的了解其对的处理:如果是以classpath*开头,它会遍历classpath.

 

    protected Resource[] findAllClassPathResources(String location) throws IOException {
        String path = location;
        if (path.startsWith("/")) {
            path = path.substring(1);
        }
        Enumeration resourceUrls = getClassLoader().getResources(path);
        Set<Resource> result = new LinkedHashSet<Resource>(16);
        while (resourceUrls.hasMoreElements()) {
            URL url = (URL) resourceUrls.nextElement();
            result.add(convertClassLoaderURL(url));
        }
        return result.toArray(new Resource[result.size()]);
    }

 http://blog.csdn.net/kkdelta/article/details/5560210,简介了在JAVA里遍历classpath中读取找到的所有符合名称的文件.

 

 

另外在加载resource的时候,其他前缀的意义如下表所示:注意classpath*只能用与指定配置文件的路径,不能用在用于getResource的参数.如appContext.getResource("classpath*:conf/bfactoryCtx.xml")会异常的.

前缀 例子 说明

classpath:

classpath:com/myapp/config.xml

从classpath中加载。

file:

file:/data/config.xml

作为 URL 从文件系统中加载。

http:

http://myserver/logo.png

作为 URL 加载。

(none)

/data/config.xml

根据 ApplicationContext 进行判断。

从Spring的源码可以看出原因:如果是classpath:开头,从classpath加载,否则尝试URL,如果失败,调用 getResourceByPath

public Resource getResource(String location) {
        Assert.notNull(location, "Location must not be null");
        if (location.startsWith(CLASSPATH_URL_PREFIX)) {
            return new ClassPathResource(location.substring(CLASSPATH_URL_PREFIX.length()), getClassLoader());
        }
        else {
            try {
                // Try to parse the location as a URL...
                URL url = new URL(location);
                return new UrlResource(url);
            }
            catch (MalformedURLException ex) {
                // No URL -> resolve as resource path.
                return getResourceByPath(location);
            }
        }
    }

 getResourceByPath会被不同ApplicationContext 实现覆盖.

如 GenericWebApplicationContext覆盖为如下:

protected Resource getResourceByPath(String path) {
        return new ServletContextResource(this.servletContext, path);
    }

如 FileSystemXmlApplicationContext覆盖为如下:

protected Resource getResourceByPath(String path) {
        if (path != null && path.startsWith("/")) {
            path = path.substring(1);
        }
        return new FileSystemResource(path);
    }

 最终从文件加载的时候仍然是JAVA中常见的读取文件的方法:

 

如ClassPathResource得到inputstream的方法是利用class loader.

    public InputStream getInputStream() throws IOException {
        InputStream is;
        if (this.clazz != null) {
            is = this.clazz.getResourceAsStream(this.path);
        }

 如FileSystemResource得到inputstream的方法是利用FileInputStream.

 

    public InputStream getInputStream() throws IOException {
        return new FileInputStream(this.file);
    }

如ServletContextResource得到inputstream的方法是利用servletContext.getResourceAsStream.

 

    public InputStream getInputStream() throws IOException {
        InputStream is = this.servletContext.getResourceAsStream(this.path);
        if (is == null) {
            throw new FileNotFoundException("Could not open " + getDescription());
        }
        return is;
    }

 

分享到:
评论

相关推荐

    spring4 中文API

    - **Application contexts and Resource paths**:讨论了应用程序上下文与资源路径的关系。 #### Validation, Data Binding, and Type Conversion - **Introduction**:概述验证、数据绑定和类型转换的基本概念。 ...

    spring2.5 学习笔记

    - **引入 Spring 的 jar 包**:添加 Spring 框架所需的依赖库到项目的 classpath 中。 - **编写测试代码**:编写简单的 Java 类来测试 Spring 是否能够正确加载和使用 Bean。 - **注意接口的使用**:确保所有的 ...

    Spring中使用classpath加载配置文件浅析

    2. **使用通配符**:当路径中包含通配符(如`**/*`)时,Spring将尝试加载所有匹配的配置文件。例如: - 如果配置文件放在bin目录下的conf文件夹内,使用通配符时,上述代码将会加载bin/conf目录及其所有子目录下...

    Spring的IoC容器(PDF)

    - **Spring中的Resource**:定义了一种资源访问策略,可以访问不同类型的资源。 - **ResourceLoader**:提供了资源加载能力,使开发者可以通过统一的方式访问各种资源。 - **ApplicationContext与ResourceLoader*...

    CXF与spring整合实现

    &lt;import resource="classpath*:META-INF/cxf/cxf-extension-soap.xml"/&gt; &lt;import resource="classpath*:META-INF/cxf/cxf-servlet.xml"/&gt; &lt;import resource="classpath:services.xml"/&gt; ``` **services.xml**文件则...

    Spring学习总结笔记

    - **WildcardClassLoader**:通过通配符加载多个配置文件,如`classpath:applicationContext-*.xml`。 5. **依赖注入(Dependency Injection, DI)** DI是Spring的核心,它通过XML配置或注解方式来实现对象间的...

    Struts2.1.6+Spring2.5.6+Hibernate3.3.1框架整合常见错误

    ### Struts2.1.6+Spring2.5.6+...2. **确认加载方式**: 如果使用`classpath*`来加载多个配置文件,则需确保所有相关配置文件都放置在正确的路径下。 #### 错误5: hibernate.cfg.xml配置问题 **错误描述**: ``` ...

    Spring通过在classpath自动扫描方式把组件纳入spring容器中管理

    5. **自动代理**: 当Spring发现一个类被`@Autowired`、`@Resource`或`@Service`等注解标记时,它会自动创建bean并处理依赖注入。对于@Controller注解的类,Spring还会处理HTTP请求映射。 6. **初始化和销毁方法**: ...

    spring整合CXF开发webService接口所需的全部jar包

    - **ApplicationContext**:Spring的应用上下文,负责加载bean定义并管理bean的生命周期。 - **Bean工厂(Bean Factory)**:Spring的基础容器,用来创建和管理bean。 - **AOP(面向切面编程)**:Spring提供了对...

    springDM开发指南(英文)

    6. **Classpath Resource Abstraction**:引入了资源抽象层,使得开发者可以通过统一的方式访问类路径上的资源,无论资源位于何处。 7. **Pluggable Extender Configuration**:Spring DM支持插件式的Extender配置...

    mybatis中文版教程

    - **整合动机**:为了使MyBatis更好地与Spring框架整合,尤其是在Spring 3.0之后的版本中,MyBatis社区推出了MyBatis-Spring项目。 - **要求**: - 熟悉Spring和MyBatis框架及其相关的术语。 - Java 5或更高版本。...

    spring 启动时加载不同的文件

    ### Spring启动时根据配置文件加载不同文件的知识点详解 #### 一、背景介绍 在实际的应用开发中,根据运行环境的不同(例如开发环境、测试环境、生产环境等),应用程序往往需要连接不同的数据库或其他资源。为了...

    spring资源文件

    4. **资源加载**:Spring使用`Resource`接口来表示资源,它可以是文件、URL、输入/输出流等。`ClassPathResource`、`FileSystemResource`和`UrlResource`是其常见实现,分别用于加载类路径、文件系统和URL路径的资源...

    spring 所有功能详解

    - **MVC与Web MVC与Spring MVC的区别**: - **MVC**是一种软件架构模式,用于构建用户界面,将应用程序分为模型、视图和控制器三个部分。 - **Web MVC**是指专为Web应用程序设计的MVC架构。 - **Spring MVC**是...

    springboot配置属性大全

    - **`spring.view.prefix`** 和 **`spring.view.suffix`**:与 `spring.mvc.view.prefix` 和 `spring.mvc.view.suffix` 类似,用于定义视图的前缀和后缀。 #### Resource 相关配置 - **`spring.resources.add-...

    spring源代码

    Resource[] resources = resolver.getResources("classpath*:META-INF/INDEX.LIST"); Assert.assertTrue(resources.length &gt; 1); //将加载多个模式匹配的Resource resources = resolver.getResources(...

    java文件的加载

    - **用途**:Spring框架提供了强大的依赖注入功能,可以轻松管理配置文件的加载。 - **示例**: ```java import org.springframework.beans.factory.BeanFactory; import org.springframework.beans.factory....

    Spring_ClassPathResource获得bean对象

    虽然这不是Spring ClassPathResource的直接功能,但你可以扩展Spring的服务,以在加载或处理资源后添加此类功能。 在实际项目中,ClassPathResource常用于加载数据库连接配置、邮件服务器设置、国际化资源文件等。...

Global site tag (gtag.js) - Google Analytics