解決SpringBoot jar包中的文件讀取問題

前言

SpringBoot微服務已成爲業界主流,從開發到部署都非常省時省力,但是最近小明開發時遇到一個問題:在代碼中讀取資源文件(比如word文檔、導出模版等),本地開發時可以正常讀取 ,但是,當我們打成jar包發佈到服務器後,再次執行程序時就會拋出找不到文件的異常。

背景

這個問題是在一次使用freemarker模版引擎導出word報告時發現的。大概說一下docx導出java實現思路:導出word的文檔格式爲docx,事先準備好一個排好版的docx文檔作爲模版,讀取解析該模版,將其中的靜態資源替換再導出。

docx文檔本身其實是一個壓縮的zip文件,將其解壓過後就會發現它有自己的目錄結構。

問題

這個docx文檔所在目錄如下圖所示:

在本地調試時,我使用如下方式讀取:

import org.springframework.util.ResourceUtils;
    public static void main(String[] args) throws IOException {
File docxTemplate = ResourceUtils.getFile("classpath:templates/docxTemplate.docx");
    }

可以正常解析使用,但是打包發佈到beta環境卻不可用。拋出異常如下:

java.io.FileNotFoundException: class path resource [templates/docxTemplate.docx] cannot be resolved to absolute file path because it does not reside in the file system: jar:file:/usr/local/subject-server.jar!/BOOT-INF/classes!/templates/docxTemplate.docx

顯而易見,這個異常告訴我們:沒有找到文件,但是將jar包解壓過後,發現這個文件是真真實實存在的。
那這到底是怎麼回事呢?這壓根難不倒我。我們要善於透過堆棧信息看本質。通過仔細觀察堆棧信息,我發現此時的文件路徑並不是一個合法的URL(文件資源定位符)。原來jar包中資源有其專門的URL形式: jar:!/{entry} )。所以,此時如果仍然按照標準的文件資源定位形式

File f=new File("jar:file:……");

定位文件,就會拋出java.io.FileNotFoundException

解決

雖然我們不能用常規操作文件的方法來讀取jar包中的資源文件docxTemplate.docx,但可以通過Class類的getResourceAsStream()方法,即通過流的方式來獲取 :

    public static void main(String[] args) throws IOException {
InputStream inputStream = WordUtil.class.getClassLoader().getResourceAsStream("templates/docxTemplate.docx");
    }

拿到流之後,就可以將其轉換爲任意一個我們需要的對象,比如FileString等等,此處我要獲取docxTemplate.docx下的目錄結構,因此我需要一個File對象,代碼舉例如下:

import org.apache.commons.io.FileUtils;

   public static void main(String[] args) throws IOException {
        InputStream inputStream = WordUtil.class.getClassLoader().getResourceAsStream("templates/docxTemplate.docx");
        File docxFile = new File("docxTemplate.docx");
        // 使用common-io的工具類即可轉換
        FileUtils.copyToFile(inputStream,docxFile);
        ZipFile zipFile = new ZipFile(docxFile);
        Enumeration<? extends ZipEntry> zipEntrys = zipFile.entries();
        // todo 記得關閉流
    }

結果

打包、發佈至beta環境,親測可用,問題完美解決。

本文可轉載,但需聲明原文出處。 程序員小明,一個很少加班的程序員。歡迎關注微信公衆號,獲取更多優質文章。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章