ResouceUtils.getFile()取不到Jar中資源文件源碼小結

来源:https://www.cnblogs.com/chyu/archive/2018/02/03/8407541.html
-Advertisement-
Play Games

Spring提供了一個工具類可以載入classpath下的文件,一般情況下無任何問題,但是當它作為公共的jar包中的工具來載入jar包中的文件時則報出找不到文件的錯誤. 點開看了一下這個工具類ResouceUtils.getFile()方法的源碼: 看了一下代碼結構簡單邏輯清晰,可能有問題的也就是上 ...


Spring提供了一個工具類可以載入classpath下的文件,一般情況下無任何問題,但是當它作為公共的jar包中的工具來載入jar包中的文件時則報出找不到文件的錯誤.

點開看了一下這個工具類ResouceUtils.getFile()方法的源碼:

public static File getFile(String resourceLocation) throws FileNotFoundException {
        Assert.notNull(resourceLocation, "Resource location must not be null");
        if (resourceLocation.startsWith(CLASSPATH_URL_PREFIX)) {
            String path = resourceLocation.substring(CLASSPATH_URL_PREFIX.length());
            String description = "class path resource [" + path + "]";
            ClassLoader cl = ClassUtils.getDefaultClassLoader();
            URL url = (cl != null ? cl.getResource(path) : ClassLoader.getSystemResource(path));
            if (url == null) {
                throw new FileNotFoundException(description +
                        " cannot be resolved to absolute file path because it does not exist");
            }
            return getFile(url, description);
        }
        try {
            // try URL
            return getFile(new URL(resourceLocation));
        }
        catch (MalformedURLException ex) {
            // no URL -> treat as file path
            return new File(resourceLocation);
        }
    }

看了一下代碼結構簡單邏輯清晰,可能有問題的也就是上圖標紅的2處.這裡我第一印象是類載入器載入資源的時候沒載入到.Debug了一下cl.getResource(path)用的類載入器是

WebAppClassLoader,想看一下內部實現,但是到這裡就跟不進去了,然後百度了一下發現這個是Jetty實現的自己的ClassLoader,截取部分關鍵的載入源碼:

public void addJars(Resource lib)  {  
    if (lib.exists() && lib.isDirectory())  
    {  
        String[] files=lib.list();
        for (int f=0;files!=null && f<files.length;f++)  {  
            try {  
                Resource fn=lib.addPath(files[f]);
                String fnlc=fn.getName().toLowerCase();  
                if (fnlc.endsWith(".jar") || fnlc.endsWith(".zip"))  {  
                    String jar=fn.toString();    
                    jar=StringUtil.replace(jar, ",", "%2C");  
                    jar=StringUtil.replace(jar, ";", "%3B");  
                    addClassPath(jar);  
                }  
            }  catch (Exception ex) {  
                Log.warn(Log.EXCEPTION,ex);  
            }  
        }  
    }  
} 

上面這塊是把jar和zip的path加到類載入器路徑中的部分源碼.繼續debug得到

URL url = (cl != null ? cl.getResource(path) : ClassLoader.getSystemResource(path));

上面的url結果:

 

實際上取到了要載入的文件路徑,由於在jar包中,協議欄位被標識為jar.到這裡看來並非類載入器導致的載入文件失敗.那隻好繼續debug往下看getFile()的源碼:

public static File getFile(URL resourceUrl, String description) throws FileNotFoundException {
        Assert.notNull(resourceUrl, "Resource URL must not be null");
        if (!URL_PROTOCOL_FILE.equals(resourceUrl.getProtocol())) {  //URL_PROTOCOL_FILE="file"
            throw new FileNotFoundException(
                    description + " cannot be resolved to absolute file path " +
                    "because it does not reside in the file system: " + resourceUrl);
        }
        try {
            return new File(toURI(resourceUrl).getSchemeSpecificPart());
        }
        catch (URISyntaxException ex) {
            // Fallback for URLs that are not valid URIs (should hardly ever happen).
            return new File(resourceUrl.getFile());
        }
    }

看到這裡有點無語了,上面紅色字體的部分實際判斷資源路徑的協議是否為file,由於在jar包中,協議是jar,故到此處直接拋出文件未找到的異常,

頓時覺得自己好傻,debug這麼久,又理了一遍類載入的過程,其實問題是很簡單的一個問題,ResouceUtils.getFile()是專門用來載入非壓縮和Jar包文件類型的資源,所以它根本不會

去嘗試載入Jar中的文件,要想載入Jar中的文件,只要用可以讀取jar中文件的方式載入即可,比如 xx.class.getClassLoader().getResouceAsStream()這種以流的形式讀取文件的方式.

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 在上一篇《Java併發系列[1] AbstractQueuedSynchronizer源碼分析之概要分析》中我們介紹了AbstractQueuedSynchronizer基本的一些概念,主要講了AQS的排隊區是怎樣實現的,什麼是獨占模式和共用模式以及如何理解結點的等待狀態。理解並掌握這些內容是後續閱 ...
  • 源代碼: "https://github.com/nnngu/LagouSpider" 效果預覽 ![][7] 思路 1、首先我們打開拉勾網,並搜索“java”,顯示出來的職位信息就是我們的目標。 2、接下來我們需要確定,怎樣將信息提取出來。 查看網頁源代碼,這時候發現,網頁源代碼裡面找不到職位相關 ...
  • 涉及內容: 函數的定義方式 函數的文字描述 空操作語句 位置參數 預設參數 關鍵參數 可變長度參數 函數的定義方式: 函數的文字說明: 為了讓別人瞭解函數的意義,或者避免自己遺忘,可以使用 字元串(不需要賦值,單引號,雙引號,多引號都行)、#註釋 將文字說明寫在函數最開始的位置 def functi... ...
  • 這裡的第一個演算法,沒什麼可以說的,一定是從最經典的冒泡演算法開始,會列出python版和c版 冒泡演算法很簡單,就是像冒泡一樣,把小的,也可以理解成輕的,從下麵浮出來 比如有個list = [3,2,5,4,1],先用3和2比,2輕,2浮上去,3沉下去,3再和5比,3比較輕,位置不變,5和4比,4浮上來 ...
  • 各位博主大大: 從這周周三開通博客到現在,已經三天了, 今天應該算是正式開始博客之路了。 先說好,本人比較懶,所以會不定時更新博客文章和隨筆哦, 還有就是大家有任何意見歡迎各種挖坑吐槽和小姐姐來討論哦,僅限純技術哦~~~~哈哈,因為小姐姐是純技術宅。 好了,先介紹自己吧,從開始寫程式到現在大概也有三 ...
  • 算術運算符: 算術運算符包含: +加法運算、 -減法運算、 *乘法運算、 /精確除法運算、 //地板除法運算、 %求餘運算、 **冪運算符 +加法運算: >>> a=1 >>> b=2 >>> c=a+b >>> c 3 -減法運算: >>> c=1000 >>> a=6000 >>> b=c-a.... ...
  • #之所以把這倆寫一起,並不是因為這倆有什麼關係,因為都太簡單,沒什麼可說的 #自定義函數的格式,def開頭,後面空格,在後面是函數名,接括弧,括弧里是入參參數 結果是1\n2\n3\n4\n5\n6\n7\n8\n9\n10\n 說明下,這裡有個遞歸,遞歸在代碼里最好不要出現,因為每遞歸一次都會占用 ...
  • 2018-02-03 還好寫過大整數運算,順利地一編A過。(這道題就是一道大整數乘法的實現代碼) 簡單的說一下吧。大整數運算的思想就是用數組儲存數字,並且依靠數組進行進位的模擬。 當然這個用的是每1個存儲一格,要想優化時間的話可以考慮每4個存儲一格。 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...