Spring ResourceLoader.getResource() & getResources()的理解


關於spring Resource的資源類型以及繼承體系我們已經在上一篇文件粗略的說了一下。Spring加載Resource文件是通過ResourceLoader來進行的,那么我們就先來看看ResourceLoader的繼承體系,讓我們對這個模塊有一個比較系統的認知。

上圖僅右邊的繼承體系,僅畫至AbstractApplicationContext,由於ApplicationContext的繼承體系,我們已經在前面章節給出,所以為了避免不必要的復雜性,本章繼承體系就不引入ApplicationContext。

  

 

 

我們還是來關注本章的重點————classpath 與 classpath*以及通配符是怎么處理的

 

首先,我們來看下ResourceLoader的源碼

 

[java]  view plain  copy
 
  1. public interface ResourceLoader {  
  2.   
  3.     /** Pseudo URL prefix for loading from the class path: "classpath:" */  
  4.     String CLASSPATH_URL_PREFIX = ResourceUtils.CLASSPATH_URL_PREFIX;  
  5.       
  6.     Resource getResource(String location);  
  7.   
  8.     ClassLoader getClassLoader();  
  9.   
  10. }  


我們發現,其實ResourceLoader接口只提供了classpath前綴的支持。而classpath*的前綴支持是在它的子接口ResourcePatternResolver中。

 

 

[java]  view plain  copy
 
  1. public interface ResourcePatternResolver extends ResourceLoader {  
  2.   
  3.     /** 
  4.      * Pseudo URL prefix for all matching resources from the class path: "classpath*:" 
  5.      * This differs from ResourceLoader's classpath URL prefix in that it 
  6.      * retrieves all matching resources for a given name (e.g. "/beans.xml"), 
  7.      * for example in the root of all deployed JAR files. 
  8.      * @see org.springframework.core.io.ResourceLoader#CLASSPATH_URL_PREFIX 
  9.      */  
  10.     String CLASSPATH_ALL_URL_PREFIX = "classpath*:";  
  11.   
  12.       
  13.     Resource[] getResources(String locationPattern) throws IOException;  
  14.   
  15. }  


   通過2個接口的源碼對比,我們發現ResourceLoader提供 classpath下單資源文件的載入,而ResourcePatternResolver提供了多資源文件的載入。

 

  ResourcePatternResolver有一個實現類:PathMatchingResourcePatternResolver,那我們直奔主題,查看PathMatchingResourcePatternResolver的getResources()

 

[java]  view plain  copy
 
  1. public Resource[] getResources(String locationPattern) throws IOException {  
  2.         Assert.notNull(locationPattern, "Location pattern must not be null");  
  3.         //是否以classpath*開頭  
  4.         if (locationPattern.startsWith(CLASSPATH_ALL_URL_PREFIX)) {  
  5.             //是否包含?或者*  
  6.             if (getPathMatcher().isPattern(locationPattern.substring(CLASSPATH_ALL_URL_PREFIX.length()))) {  
  7.                 // a class path resource pattern  
  8.                 return findPathMatchingResources(locationPattern);  
  9.             }  
  10.             else {  
  11.                 // all class path resources with the given name  
  12.                 return findAllClassPathResources(locationPattern.substring(CLASSPATH_ALL_URL_PREFIX.length()));  
  13.             }  
  14.         }  
  15.         else {  
  16.             // Only look for a pattern after a prefix here  
  17.             // (to not get fooled by a pattern symbol in a strange prefix).  
  18.             int prefixEnd = locationPattern.indexOf(":") + 1;  
  19.             //是否包含?或者*  
  20.             if (getPathMatcher().isPattern(locationPattern.substring(prefixEnd))) {  
  21.                 // a file pattern  
  22.                 return findPathMatchingResources(locationPattern);  
  23.             }  
  24.             else {  
  25.                 // a single resource with the given name  
  26.                 return new Resource[] {getResourceLoader().getResource(locationPattern)};  
  27.             }  
  28.         }  
  29.     }  


由此我們可以看出在加載配置文件時,以是否是以classpath*開頭分為2大類處理場景,每大類在又根據路徑中是否包括通配符分為2小類進行處理,

 

處理的流程圖如下:

從上圖看,整個加載資源的場景有三條處理流程

 

  • 以classpath*開頭,但路徑不包含通配符的
             讓我們來看看findAllClassPathResources是怎么處理的
[java]  view plain  copy
 
  1. protected Resource[] findAllClassPathResources(String location) throws IOException {  
  2.     String path = location;  
  3.     if (path.startsWith("/")) {  
  4.         path = path.substring(1);  
  5.     }  
  6.     Enumeration<URL> resourceUrls = getClassLoader().getResources(path);  
  7.     Set<Resource> result = new LinkedHashSet<Resource>(16);  
  8.     while (resourceUrls.hasMoreElements()) {  
  9.         URL url = resourceUrls.nextElement();  
  10.         result.add(convertClassLoaderURL(url));  
  11.     }  
  12.     return result.toArray(new Resource[result.size()]);  
  13. }  

    我們可以看到,最關鍵的一句代碼是:Enumeration<URL> resourceUrls = getClassLoader().getResources(path); 
[java]  view plain  copy
 
  1.     public ClassLoader getClassLoader() {  
  2.         return getResourceLoader().getClassLoader();  
  3.     }  
  4.   
  5.   
  6. public ResourceLoader getResourceLoader() {  
  7.         return this.resourceLoader;  
  8.     }  
  9.   
  10. //默認情況下  
  11. public PathMatchingResourcePatternResolver() {  
  12.         this.resourceLoader = new DefaultResourceLoader();  
  13.     }  
其實上面這3個方法不是最關鍵的,之所以貼出來,是讓大家清楚整個調用鏈,其實這種情況最關鍵的代碼在於ClassLoader的getResources()方法。那么我們同樣跟進去,看看源碼
[java]  view plain  copy
 
  1. public Enumeration<URL> getResources(String name) throws IOException {  
  2. Enumeration[] tmp = new Enumeration[2];  
  3. if (parent != null) {  
  4.     tmp[0] = parent.getResources(name);  
  5. else {  
  6.     tmp[0] = getBootstrapResources(name);  
  7. }  
  8. tmp[1] = findResources(name);  
  9.   
  10. return new CompoundEnumeration(tmp);  
  11.    }  
是不是一目了然了?當前類加載器,如果存在父加載器,則向上迭代獲取資源, 因此能加到jar包里面的資源文件。

  • 不以classpath*開頭,且路徑不包含通配符的
處理邏輯如下           
[java]  view plain  copy
 
  1. return new Resource[] {getResourceLoader().getResource(locationPattern)};  
上面我們已經貼過getResourceLoader()的邏輯了, 即默認是DefaultResourceLoader(),那我們進去看看getResouce()的實現
[java]  view plain  copy
 
  1. public Resource getResource(String location) {  
  2.     Assert.notNull(location, "Location must not be null");  
  3.     if (location.startsWith(CLASSPATH_URL_PREFIX)) {  
  4.         return new ClassPathResource(location.substring(CLASSPATH_URL_PREFIX.length()), getClassLoader());  
  5.     }  
  6.     else {  
  7.         try {  
  8.             // Try to parse the location as a URL...  
  9.             URL url = new URL(location);  
  10.             return new UrlResource(url);  
  11.         }  
  12.         catch (MalformedURLException ex) {  
  13.             // No URL -> resolve as resource path.  
  14.             return getResourceByPath(location);  
  15.         }  
  16.     }  
  17. }  

其實很簡單,如果以classpath開頭,則創建為一個ClassPathResource,否則則試圖以URL的方式加載資源,創建一個UrlResource.
  • 路徑包含通配符的
             這種情況是最復雜的,涉及到層層遞歸,那我把加了注釋的代碼發出來大家看一下,其實主要的思想就是
1.先獲取目錄,加載目錄里面的所有資源
2.在所有資源里面進行查找匹配,找出我們需要的資源
[java]  view plain  copy
 
  1. protected Resource[] findPathMatchingResources(String locationPattern) throws IOException {  
  2.         //拿到能確定的目錄,即拿到不包括通配符的能確定的路徑  比如classpath*:/aaa/bbb/spring-*.xml 則返回classpath*:/aaa/bbb/                                     //如果是classpath*:/aaa/*/spring-*.xml,則返回 classpath*:/aaa/  
  3.         String rootDirPath = determineRootDir(locationPattern);  
  4.         //得到spring-*.xml  
  5.         String subPattern = locationPattern.substring(rootDirPath.length());  
  6.         //遞歸加載所有的根目錄資源,要注意的是遞歸的時候又得考慮classpath,與classpath*的情況,而且還得考慮根路徑中是否又包含通配符,參考上面那張流程圖  
  7.         Resource[] rootDirResources = getResources(rootDirPath);  
  8.         Set<Resource> result = new LinkedHashSet<Resource>(16);  
  9.         //將根目錄所有資源中所有匹配我們需要的資源(如spring-*)加載result中  
  10.         for (Resource rootDirResource : rootDirResources) {  
  11.             rootDirResource = resolveRootDirResource(rootDirResource);  
  12.             if (isJarResource(rootDirResource)) {  
  13.                 result.addAll(doFindPathMatchingJarResources(rootDirResource, subPattern));  
  14.             }  
  15.             else if (rootDirResource.getURL().getProtocol().startsWith(ResourceUtils.URL_PROTOCOL_VFS)) {  
  16.                 result.addAll(VfsResourceMatchingDelegate.findMatchingResources(rootDirResource, subPattern, getPathMatcher()));  
  17.             }  
  18.             else {  
  19.                 result.addAll(doFindPathMatchingFileResources(rootDirResource, subPattern));  
  20.             }  
  21.         }  
  22.         if (logger.isDebugEnabled()) {  
  23.             logger.debug("Resolved location pattern [" + locationPattern + "] to resources " + result);  
  24.         }  
  25.         return result.toArray(new Resource[result.size()]);  
  26.     }  
 
值得注解一下的是determineRootDir()方法的作用,是確定根目錄,這個根目錄必須是一個能確定的路徑,不會包含通配符。如果classpath*:aa/bb*/spring-*.xml,得到的將是classpath*:aa/  可以看下他的源碼
 
[java]  view plain  copy
 
  1. protected String determineRootDir(String location) {  
  2.     int prefixEnd = location.indexOf(":") + 1;  
  3.     int rootDirEnd = location.length();  
  4.     while (rootDirEnd > prefixEnd && getPathMatcher().isPattern(location.substring(prefixEnd, rootDirEnd))) {  
  5.         rootDirEnd = location.lastIndexOf('/', rootDirEnd - 2) + 1;  
  6.     }  
  7.     if (rootDirEnd == 0) {  
  8.         rootDirEnd = prefixEnd;  
  9.     }  
  10.     return location.substring(0, rootDirEnd);  
  11. }  
 



分析到這,結合測試我們可以總結一下:
1.無論是classpath還是classpath*都可以加載整個classpath下(包括jar包里面)的資源文件。
2.classpath只會返回第一個匹配的資源,查找路徑是優先在項目中存在資源文件,再查找jar包。
3.文件名字包含通配符資源(如果spring-*.xml,spring*.xml),   如果根目錄為"", classpath加載不到任何資源, 而classpath*則可以加載到classpath中可以匹配的目錄中的資源,但是不能加載到jar包中的資源
    
      第1,2點比較好表理解,大家可以自行測試,第三點表述有點繞,舉個例,現在有資源文件結構如下:
 
classpath:notice*.txt                                                               加載不到資源
classpath*:notice*.txt                                                            加載到resource根目錄下notice.txt
classpath:META-INF/notice*.txt                                          加載到META-INF下的一個資源(classpath是加載到匹配的第一個資源,就算刪除classpath下的notice.txt,他仍然可以                                                                                                  加載jar包中的notice.txt)
classpath:META-*/notice*.txt                                              加載不到任何資源
classpath*:META-INF/notice*.txt                                        加載到classpath以及所有jar包中META-INF目錄下以notice開頭的txt文件
classpath*:META-*/notice*.txt                                             只能加載到classpath下 META-INF目錄的notice.txt


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM