本篇文章是由朋友的一篇博客引出的,博客原文地址:http://jinnianshilongnian.iteye.com/blog/1416322
他这篇博客比较细的讲解了classpath与classpath*,以及通配符的使用,那些配置能成功加载到资源,那些配置加载不了资源。但是我相信仍然有很多同学不明白,为什么是这样的,知其然,不知其所以然,那么本篇文章将慢慢为你揭开神秘的面纱,让你知其然,更知其所以然。
关于Spring Resource的资源类型以及继承体系我们已经在上一篇文件粗略的说了一下。Spring加载Resource文件是通过ResourceLoader来进行的,那么我们就先来看看ResourceLoader的继承体系,让我们对这个模块有一个比较系统的认知。
上图仅右边的继承体系,仅画至AbstractApplicationContext,由于ApplicationContext的继承体系,我们已经在前面章节给出,所以为了避免不必要的复杂性,本章继承体系就不引入ApplicationContext。
我们还是来关注本章的重点————classpath 与 classpath*以及通配符是怎么处理的
首先,我们来看下ResourceLoader的源码
public interface ResourceLoader { /** Pseudo URL prefix for loading from the class path: "classpath:" */ String CLASSPATH_URL_PREFIX = ResourceUtils.CLASSPATH_URL_PREFIX; Resource getResource(String location); ClassLoader getClassLoader(); }
我们发现,其实ResourceLoader接口只提供了classpath前缀的支持。而classpath*的前缀支持是在它的子接口ResourcePatternResolver中。
public interface ResourcePatternResolver extends ResourceLoader { /** * Pseudo URL prefix for all matching resources from the class path: "classpath*:" * This differs from ResourceLoader's classpath URL prefix in that it * retrieves all matching resources for a given name (e.g. "/beans.xml"), * for example in the root of all deployed JAR files. * @see org.springframework.core.io.ResourceLoader#CLASSPATH_URL_PREFIX */ String CLASSPATH_ALL_URL_PREFIX = "classpath*:"; Resource[] getResources(String locationPattern) throws IOException; }
通过2个接口的源码对比,我们发现ResourceLoader提供 classpath下单资源文件的载入,而ResourcePatternResolver提供了多资源文件的载入。
ResourcePatternResolver有一个实现类:PathMatchingResourcePatternResolver,那我们直奔主题,查看PathMatchingResourcePatternResolver的getResources()
public Resource[] getResources(String locationPattern) throws IOException { Assert.notNull(locationPattern, "Location pattern must not be null"); //是否以classpath*开头 if (locationPattern.startsWith(CLASSPATH_ALL_URL_PREFIX)) { //是否包含?或者* if (getPathMatcher().isPattern(locationPattern.substring(CLASSPATH_ALL_URL_PREFIX.length()))) { // a class path resource pattern return findPathMatchingResources(locationPattern); } else { // all class path resources with the given name return findAllClassPathResources(locationPattern.substring(CLASSPATH_ALL_URL_PREFIX.length())); } } else { // Only look for a pattern after a prefix here // (to not get fooled by a pattern symbol in a strange prefix). int prefixEnd = locationPattern.indexOf(":") + 1; //是否包含?或者* if (getPathMatcher().isPattern(locationPattern.substring(prefixEnd))) { // a file pattern return findPathMatchingResources(locationPattern); } else { // a single resource with the given name return new Resource[] {getResourceLoader().getResource(locationPattern)}; } } }
由此我们可以看出在加载配置文件时,以是否是以classpath*开头分为2大类处理场景,每大类在又根据路径中是否包括通配符分为2小类进行处理,
处理的流程图如下:
从上图看,整个加载资源的场景有三条处理流程
- 以classpath*开头,但路径不包含通配符的
protected Resource[] findAllClassPathResources(String location) throws IOException { String path = location; if (path.startsWith("/")) { path = path.substring(1); } Enumeration<URL> resourceUrls = getClassLoader().getResources(path); Set<Resource> result = new LinkedHashSet<Resource>(16); while (resourceUrls.hasMoreElements()) { URL url = resourceUrls.nextElement(); result.add(convertClassLoaderURL(url)); } return result.toArray(new Resource[result.size()]); }我们可以看到,最关键的一句代码是:Enumeration<URL> resourceUrls = getClassLoader().getResources(path);
public ClassLoader getClassLoader() { return getResourceLoader().getClassLoader(); } public ResourceLoader getResourceLoader() { return this.resourceLoader; } //默认情况下 public PathMatchingResourcePatternResolver() { this.resourceLoader = new DefaultResourceLoader(); }其实上面这3个方法不是最关键的,之所以贴出来,是让大家清楚整个调用链,其实这种情况最关键的代码在于ClassLoader的getResources()方法。那么我们同样跟进去,看看源码
public Enumeration<URL> getResources(String name) throws IOException { Enumeration[] tmp = new Enumeration[2]; if (parent != null) { tmp[0] = parent.getResources(name); } else { tmp[0] = getBootstrapResources(name); } tmp[1] = findResources(name); return new CompoundEnumeration(tmp); }是不是一目了然了?当前类加载器,如果存在父加载器,则向上迭代获取资源, 因此能加到jar包里面的资源文件。
- 不以classpath*开头,且路径不包含通配符的
return new Resource[] {getResourceLoader().getResource(locationPattern)};上面我们已经贴过getResourceLoader()的逻辑了, 即默认是DefaultResourceLoader(),那我们进去看看getResouce()的实现
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); } } }其实很简单,如果以classpath开头,则创建为一个ClassPathResource,否则则试图以URL的方式加载资源,创建一个UrlResource.
- 路径包含通配符的
protected Resource[] findPathMatchingResources(String locationPattern) throws IOException { //拿到能确定的目录,即拿到不包括通配符的能确定的路径 比如classpath*:/aaa/bbb/spring-*.xml 则返回classpath*:/aaa/bbb/ //如果是classpath*:/aaa/*/spring-*.xml,则返回 classpath*:/aaa/ String rootDirPath = determineRootDir(locationPattern); //得到spring-*.xml String subPattern = locationPattern.substring(rootDirPath.length()); //递归加载所有的根目录资源,要注意的是递归的时候又得考虑classpath,与classpath*的情况,而且还得考虑根路径中是否又包含通配符,参考上面那张流程图 Resource[] rootDirResources = getResources(rootDirPath); Set<Resource> result = new LinkedHashSet<Resource>(16); //将根目录所有资源中所有匹配我们需要的资源(如spring-*)加载result中 for (Resource rootDirResource : rootDirResources) { rootDirResource = resolveRootDirResource(rootDirResource); if (isJarResource(rootDirResource)) { result.addAll(doFindPathMatchingJarResources(rootDirResource, subPattern)); } else if (rootDirResource.getURL().getProtocol().startsWith(ResourceUtils.URL_PROTOCOL_VFS)) { result.addAll(VfsResourceMatchingDelegate.findMatchingResources(rootDirResource, subPattern, getPathMatcher())); } else { result.addAll(doFindPathMatchingFileResources(rootDirResource, subPattern)); } } if (logger.isDebugEnabled()) { logger.debug("Resolved location pattern [" + locationPattern + "] to resources " + result); } return result.toArray(new Resource[result.size()]); }值得注解一下的是determineRootDir()方法的作用,是确定根目录,这个根目录必须是一个能确定的路径,不会包含通配符。如果classpath*:aa/bb*/spring-*.xml,得到的将是classpath*:aa/ 可以看下他的源码
protected String determineRootDir(String location) { int prefixEnd = location.indexOf(":") + 1; int rootDirEnd = location.length(); while (rootDirEnd > prefixEnd && getPathMatcher().isPattern(location.substring(prefixEnd, rootDirEnd))) { rootDirEnd = location.lastIndexOf('/', rootDirEnd - 2) + 1; } if (rootDirEnd == 0) { rootDirEnd = prefixEnd; } return location.substring(0, rootDirEnd); }分析到这,结合测试我们可以总结一下:
相关推荐
本文将详细分析Spring通过classpath加载配置文件的不同情形,并提供解决加载过程中可能遇到的问题的方法。 首先,我们来看一下Spring加载配置文件的基本方式。在Spring中,可以通过ApplicationContext接口的实现类...
通过对这些示例的分析,我们可以深入理解Spring 2.5中的自动扫描classpath是如何工作的,以及它如何与AOP、数据源管理、依赖注入、bean作用域等核心概念相结合,构建出高效灵活的Java应用。通过阅读源码和实践这些...
#### 三、案例分析与实现 **1. 配置文件设计** - **deploy.properties**: 这个文件包含了一个名为`DATE_SOURCE`的属性,用于指示加载哪个数据库连接配置文件。 - 当`DATE_SOURCE`为`0`时,加载`dbconn.properties`...
通过以上内容的详细解读,我们可以看出Spring Boot 1.4.x版本不仅提供了强大的自动化配置能力,还极大地简化了应用的部署和维护过程。这对于想要快速搭建微服务架构的开发者来说是非常有利的。同时,Spring Boot也为...
- 可以通过ResourceLoader接口来统一管理资源的加载过程。 #### 三、Spring框架的实际应用案例 - **轻量级J2EE结构的案例分析** - 通过综合运用Spring与其他框架(如Struts、WebWork2、Hibernate等),构建出高...
在深入探讨Spring Boot的类加载过程之前,让我们先理解一下类加载的基本概念。Java中的类加载是由JVM(Java虚拟机)执行的,它负责将类的字节码加载到内存中,进行校验、解析,并准备类的数据。在Spring Boot中,这...
- **原因分析**: - 文件或资源未放置在正确的路径下。 - 类路径设置错误。 - **解决方案**: - 检查资源文件的路径是否正确。例如,在`applicationContext-hibernate.xml`文件中的`mappingResources`属性应使用...
- **背景与意义**:在传统的 Spring 应用开发过程中,需要进行大量的手动配置,这不仅耗时且容易出错。 - **Spring Boot 解决方案**:Spring Boot 通过提供一系列的默认配置和约定,极大地减少了配置的工作量,使得...
- **加密和解密**:提供了关于如何使用Spring Cloud Config加密和解密配置属性的指南。 - **密钥管理**:解释了如何管理加密密钥,包括密钥的存储和轮换策略。 #### Spring Cloud Bus - **推送通知和Spring Cloud ...
在使用 Spring 框架时,可能会遇到加载 XSD 文件发生错误的问题,本文将对此问题进行详细的分析和解决。 问题描述 在使用 Spring 框架时,可能会出现 org.xml.sax.SAXParseException 异常,错误信息为 schema_...
在Spring框架的使用过程中,有时候我们可能会遇到"spring没有提示"的情况,这通常指的是Spring的自动提示、自动配置或者智能感知功能没有正常工作。这里我们将深入探讨Spring框架的一些核心概念和可能遇到的问题,...
1. **检查配置文件路径**: 确认Spring配置文件中关于`SessionFactory`的配置是否正确,如路径是否正确。 2. **检查依赖库**: 确认项目中是否包含了正确的版本的Struts2、Spring和Hibernate等依赖库。 3. **检查日志...
- **RxJava与Spring MVC**:讲解如何在Spring MVC中集成RxJava。 - **指标:Spectator,Servo和Atlas**: - **维度与层次度量**:介绍如何使用维度和层次结构来组织度量。 - **默认度量集合**:说明Spring Cloud...
1. **需求分析**: - 用户注册程序是一个典型的例子,演示了如何使用 Struts 2 + Spring 进行开发。 - 程序的主要功能包括:收集用户输入的信息(用户名、密码等),验证这些信息,并将有效数据存储到数据库中。 ...
(3)**配置 Spring**:创建 Spring 的配置文件(如 `beans.xml`),定义数据源、SqlSessionFactory 和 SqlSessionTemplate 的 Bean,同时配置 MyBatis 的扫描路径,以便 Spring 自动加载 Mapper 文件。 (4)**...
### Spring Boot概述与核心知识点详解 #### Spring Boot简介 Spring Boot 是由 Pivotal 团队提供的全新框架,其设计目标是为了简化新Spring应用的初始搭建以及开发过程。该框架使用了特定的方式(比如自动配置)来...
【标题】"MyEclipse for Spring Demo Project" 是一个基于MyEclipse集成开发环境和Spring框架的示例项目,旨在帮助开发者了解如何在MyEclipse中配置和开发Spring应用程序。这个项目展示了如何利用MyEclipse的工具集...
### Spring与Hibernate...通过上述分析可以看出,合理的配置不仅可以简化开发过程,还能有效提高应用的稳定性和性能。在实际项目中,应根据具体需求灵活选择合适的配置方式,并不断探索和优化,以达到最佳的应用效果。
通过对这些文件的分析和代码的学习,你可以逐步理解Spring框架的工作原理,掌握如何在实际项目中应用Spring进行开发。记得动手实践,多写代码,结合官方文档和教程,你的Spring技能将会得到显著提升。