SpringBoot下載文件遇到文件損壞等問(wèn)題解決方案
問(wèn)題一:下載的文件名稱出現(xiàn)中文亂碼的問(wèn)題
解決方案:
response.setHeader("Content-Disposition", "attachment;filename=" + new String("下載模板".getBytes("UTF-8"), "ISO8859-1"));
問(wèn)題二:在swagger中測(cè)試下載接口,點(diǎn)擊下載的文件,發(fā)現(xiàn)文件名是亂碼的問(wèn)題
解決方案:
response.setHeader("Content-Disposition", "attachment;fileName=" + URLEncoder.encode("線索導(dǎo)入模板.xlsx","utf8"));
說(shuō)明:通過(guò)URLEncoder.encode
函數(shù)對(duì)文件名稱處理后,無(wú)論是在瀏覽器調(diào)用GET請(qǐng)求下載文件,還是Swagger中調(diào)用下載接口,都不會(huì)出現(xiàn)文件名亂碼問(wèn)題。
問(wèn)題三:下載的excel文件打開(kāi)時(shí)總是提示部分內(nèi)容有問(wèn)題,嘗試恢復(fù)。
問(wèn)題原因:
一般有2種情況:
1、由于沒(méi)有找到文件,下載的文件字節(jié)大小為0,這種情況文件完全打不開(kāi)
2、讀取的文件大小和元素文件的大小不一致,這種情況會(huì)提升自動(dòng)修復(fù)。
解決辦法:
網(wǎng)上最多的解決方案是主動(dòng)在response的Header中設(shè)置Content-Length大小。但這種方式其實(shí)是錯(cuò)誤的。文件的Content-Length其實(shí)可以從返回流中直接獲取,并不需要用戶主動(dòng)去設(shè)置。這里的問(wèn)題核心應(yīng)該是思考:為什么下載的文件和元素文件的大小會(huì)不一致?
下面的2個(gè)獲取inputStream的長(zhǎng)度的API,只有在讀取磁盤上具體文件中才比較適用。如果是jar包中的文件,是獲取不到大小。
加上設(shè)置大小:
response.addHeader("Content-Length",String.valueOf(file.length())); //response.addHeader("Content-Length",String.valueOf(inputStream.available()));
問(wèn)題四:采用BufferedInputStream緩沖流讀寫文件導(dǎo)致輸出文件和原始文件體積差異的問(wèn)題
由于下載的文件體積總是比元素文件體積大一點(diǎn)點(diǎn),導(dǎo)致文件打開(kāi)提示異常修復(fù)。
outputStream = response.getOutputStream(); bis = new BufferedInputStream(inputStream); //緩沖數(shù)組,每次讀取1024 byte[] buff = new byte[1024]; while (bis.read(buff)!= -1) { //異常代碼行 outputStream.write(buff, 0, buff.length); } outputStream.flush();
原因分析:
出現(xiàn)問(wèn)題的原因就是buff.length,數(shù)組聲明后長(zhǎng)度就是固定的,而不是獲取里面讀取的內(nèi)容的字節(jié)長(zhǎng)度,所以導(dǎo)致這里的buff.length的值始終是1024。
解決方案:
outputStream = response.getOutputStream(); bis = new BufferedInputStream(inputStream); //緩沖流,每次讀取1024 byte[] buff = new byte[1024]; int readLength = 0; while (( readLength = bis.read(buff)) != -1) { //每次寫入緩沖流buff讀到的字節(jié)長(zhǎng)度,而不是buff.length outputStream.write(buff, 0, readLength); } outputStream.flush();
問(wèn)題五:開(kāi)發(fā)環(huán)境下載成功,打成jar包發(fā)布到服務(wù)器上部署就出現(xiàn)下載失敗問(wèn)題
原因:
Resource下的文件是存在于jar這個(gè)文件里面,在磁盤上是沒(méi)有真實(shí)路徑存在的,它其實(shí)是位于jar內(nèi)部的一個(gè)路徑。所以通過(guò)ResourceUtils.getFile
或者this.getClass().getResource("")方法無(wú)法正確獲取文件。
解決:
通過(guò)ClassPathResource
讀取文件流
InputStream inputStream = getClass().getClassLoader().getResourceAsStream("template/template.xlsx");
下載方式
方式一:經(jīng)典的緩沖流BufferedInputStream讀取法,一般讀取比較大的文件,優(yōu)先考慮緩沖流讀取方式
方式二:利用spring的FileCopyUtils工具類,小文件優(yōu)先考慮此方式,代碼更簡(jiǎn)單不易出錯(cuò)。
實(shí)例
1、控制層代碼 @Operation(summary = "下載模版",description = "下載模版") @GetMapping("/download") public void download(HttpServletResponse response){ templateService.download(response); }
/** * 下載線索模板 * @param response */ public void download(HttpServletResponse response) { InputStream inputStream = null; BufferedInputStream bis = null; OutputStream outputStream = null; try { inputStream=getClass().getClassLoader().getResourceAsStream("template/template.xlsx"); response.setContentType("application/octet-stream"); response.setHeader("content-type", "application/octet-stream"); //待下載文件名 String fileName = URLEncoder.encode("模板.xlsx","utf8"); response.setHeader("Content-Disposition", "attachment;fileName=" + fileName); outputStream = response.getOutputStream(); //加上設(shè)置大小 下載下來(lái)的excel文件才不會(huì)在打開(kāi)前提示修復(fù) //這里流的長(zhǎng)度很難在開(kāi)始讀取前獲取,特別是打成jar包后,讀取inputStream長(zhǎng)度經(jīng)常失敗 //response.addHeader("Content-Length",String.valueOf(classPathResource.getFile().length())); //response.addHeader("Content-Length",String.valueOf(inputStream.available())); //方式一:經(jīng)典的緩沖流BufferedInputStream讀取法 // bis = new BufferedInputStream(inputStream); //緩沖流,每次讀取1024 // byte[] buff = new byte[1024]; // int readLength = 0; // while (( readLength = bis.read(buff)) != -1) { // outputStream.write(buff, 0, readLength); // } // outputStream.flush(); //方式二:利用spring的FileCopyUtils工具類 if(inputStream!=null){ byte[] results = FileCopyUtils.copyToByteArray(inputStream); outputStream.write(results); outputStream.flush(); } } catch ( IOException e ) { log.error("文件下載失敗,e"); } finally { IOUtils.closeQuietly(outputStream); IOUtils.closeQuietly(inputStream); IOUtils.closeQuietly(bis); } }
以上就是SpringBoot下載文件遇到文件損壞等問(wèn)題解決方案的詳細(xì)內(nèi)容,更多關(guān)于SpringBoot下載文件的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
詳解Java的堆內(nèi)存與棧內(nèi)存的存儲(chǔ)機(jī)制
這篇文章主要介紹了Java的堆內(nèi)存與棧內(nèi)存的存儲(chǔ)機(jī)制,包括JVM的內(nèi)存優(yōu)化和GC等相關(guān)方面內(nèi)容,需要的朋友可以參考下2016-01-01Java注冊(cè)郵箱激活驗(yàn)證實(shí)現(xiàn)代碼
這篇文章主要介紹了Java注冊(cè)郵箱激活驗(yàn)證實(shí)現(xiàn)代碼,有需要的朋友可以參考一下2013-12-12Eclipse新建項(xiàng)目不可選擇Java Project問(wèn)題解決方案
這篇文章主要介紹了Eclipse新建項(xiàng)目不可選擇Java Project問(wèn)題解決方案,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-07-07springboot使用jasypt對(duì)配置文件加密加密數(shù)據(jù)庫(kù)連接的操作代碼
這篇文章主要介紹了springboot使用jasypt對(duì)配置文件加密加密數(shù)據(jù)庫(kù)連接的操作代碼,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2024-01-01