Spring中的classpath与classpath*一直是开发中的心病,没有时间静下心来研究下,终于一气呵成!网上总结的也比较多,各种各样的说法,还不如自己亲自解读一下spring中的源码,这样下次再次使用心里就安心多了,欢迎支持!

一、问题描述

使用spring时import资源文件时路径查找顺序不明(开发中的疑惑),或者加载资源失败(不知道怎么更改路径)?

二、classpath代码解析

    直接上代码,这是代码中的主类 PathMatchingResourcePatternResolver 类,位于包 org.springframework.core.io.support 下,该类中的 getResources 函数是逻辑的核心,如下: 


public Resource[] getResources(String locationPattern)throws IOException {   
    if (locationPattern.startsWith(CLASSPATH_ALL_URL_PREFIX)) {  
        if (getPathMatcher().isPattern(locationPattern.substring(CLASSPATH_ALL_URL_PREFIX.length()))) {  
            // case 1:如果以classpath*开头且包含?或者* ,例如查找: classpath*: applicationContext-*.xml
            return findPathMatchingResources(locationPattern);  
        }  
        else {  
            // case 2: 不包含?或者*,直接全名查找,例如查找: classpath*: applicationContext-test.xml
            return findAllClassPathResources(locationPattern.substring(CLASSPATH_ALL_URL_PREFIX.length()));  
        }  
    }  
    else {
        // 以 classpath:开头
        int prefixEnd = locationPattern.indexOf(":") +1;  
        if (getPathMatcher().isPattern(locationPattern.substring(prefixEnd))) {  
            // case 3: 如果不是以classpath*开头且包含?或者* ,例如查找: classpath: applicationContext-*.xml
            return findPathMatchingResources(locationPattern);  
        }  
        else {  
            // case 4: 如果不是以classpath*开头且不包含?或者* ,例如查找: classpath: applicationContext-test.xml  
            returnnew Resource[] {getResourceLoader().getResource(locationPattern)};  
        }  
    }  
}




2.1 如果以classpath*开头且包含?或者*

例如查找: classpath*: applicationContext-*.xml ,使用findPathMatchingResources函数,看下该函数:

protected Resource[] findPathMatchingResources(String locationPattern)throws IOException {
    /*函数determineRootDir函数是拿到能够确定的目录,如 classpath*:/aaa/bbb/applicationContext-*.xml 则返回classpath*:/aaa/bbb/
                                                   classpath*:/aaa/*/applicationContext-*.xml,则返回 classpath*:/aaa/ 
    (代码就不再粘贴了,有兴趣的可以看下源码)*/
    String rootDirPath = determineRootDir(locationPattern);
    // 获取字符串locationPattern中后面不确定的内容
    String subPattern = locationPattern.substring(rootDirPath.length());
    // 递归加载已经确定的内容
    Resource[] rootDirResources = getResources(rootDirPath);
    Set result = new LinkedHashSet<>(16);
    for (Resource rootDirResource : rootDirResources) {
        rootDirResource = resolveRootDirResource(rootDirResource);
        URL rootDirURL = rootDirResource.getURL();
        if (equinoxResolveMethod !=null) {
            if (rootDirURL.getProtocol().startsWith("bundle")) {
                rootDirURL = (URL) ReflectionUtils.invokeMethod(equinoxResolveMethod,null, rootDirURL);
                rootDirResource = new UrlResource(rootDirURL);
            }
        }
        if (rootDirURL.getProtocol().startsWith(ResourceUtils.URL_PROTOCOL_VFS)) {
            // is general JBoss VFS resource: "vfs" 
            result.addAll(VfsResourceMatchingDelegate.findMatchingResources(rootDirURL, subPattern, getPathMatcher()));
        }
        elseif (ResourceUtils.isJarURL(rootDirURL) || isJarResource(rootDirResource)) {
            // is zip, jar, wsjar or vfszip的一种就加载
            result.addAll(doFindPathMatchingJarResources(rootDirResource, rootDirURL, subPattern));
        }
        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()]);
}



                 

总体来说:该函数把locationPattern拆分成两部分:rootDirPath 和subPattern,rootDirPath是根目录路径,subPattern是子目录路径匹配规则字符串。遍历根目录下的所有子目录、并得到所有的子目录在doFindPathMatchingFileResources(rootDirResource, subPattern)方法中,再根据子目录逐个逐个去匹配subPattern。


2.2 如果以classpath*开头且不包含?或者*,直接全名查找

例如查找: classpath*: applicationContext-test.xml,使用findAllClassPathResources函数,看下该函数:

protected Resource[] findAllClassPathResources(String location)throws IOException {
    String path = location;
    if (path.startsWith("/")) {
        path = path.substring(1);
    }
    Set result = doFindAllClassPathResources(path);
    if (logger.isDebugEnabled()) {
        logger.debug("Resolved classpath location [" + location +"] to resources " + result);
    }
    return result.toArray(new Resource[result.size()]);
}

其实核心在doFindAllClassPathResources,该函数扫描该路径下的所有资料,其函数体如下:

protected Set doFindAllClassPathResources(String path) throws IOException {
    Set result= new LinkedHashSet<>(16);
    ClassLoader cl = getClassLoader();
    Enumeration resourceUrls = (cl!= null? cl.getResources(path) : ClassLoader.getSystemResources(path));
    while (resourceUrls.hasMoreElements()) {
        URL url = resourceUrls.nextElement();
        result.add(convertClassLoaderURL(url));
    }
    if ("".equals(path)) {
        // 如果路径为空的话,就找到所有jar包,加到result中(该函数不再深入)
        addAllClassLoaderJarRoots(cl, result);
    }
    return result;
}



                          

在该函数中重要:

Enumeration resourceUrls = (cl != null ? cl.getResources(path) : ClassLoader.getSystemResources(path));

即如果当前类加载器cl不为空的话,就调用cl.getResources(path),cl为空的话就使用系统类加载器去加载,事实上cl.getResources(path)内容使用的即是双亲委派模型(当前类加载器加载资源,如果存在父加载器,则用父加载器进行加载)。


2.3 如果不是以classpath*开头且包含?或者*

例如查找: classpath: applicationContext-*.xml,使用函数findPathMatchingResources,类似2.1情况。

2.4 如果不是以classpath*开头且不包含?或者*

例如查找: classpath: applicationContext-test.xml,直接“getResourceLoader().getResource(locationPattern)”,即直接使用当然的资源加载器去加载,这里默认使用的是DefaultResourceLoader()。

三、总结及测试

写了个小测试,测试代码如下:

private staticvoid output(String location)throws Exception {
    ResourcePatternResolver resourceLoader = new PathMatchingResourcePatternResolver();
    Resource[] source = resourceLoader.getResources(location);
//        System.out.println("source.size: " + source.length);
    for (int i =0; i < source.length; i++) {
        Resource resource = source[i];
        System.out.println(resource);

    }
}

很多个测试结果如下图: 

mat 指定java路径 maven classpath路径_xml