SpringBoot下載文件的正確解法方式
前言
最近遇到一個(gè)奇怪的需求,前端通過post請(qǐng)求下載壓縮文件,同時(shí)會(huì)傳給后端一些數(shù)據(jù),用于生成壓縮包。此時(shí)后端接口就不僅僅是生成壓縮文件流輸出給前端。而必須要有報(bào)錯(cuò)能力與異常處理能力。即如果后端報(bào)錯(cuò),前端應(yīng)該是下載不了文件流。
比較一般的解法
一般而言,Spring Boot生成文件流供前端下載,會(huì)直接將文件流寫入到 HttpServletResponse.getOutputStream()
,然而這樣會(huì)有一個(gè)問題,無論后端如何報(bào)錯(cuò),前端都能成功下載文件,因?yàn)?nbsp;status=200
。即如下寫法:
@PostMapping(value = "/project/code/download") public void downloadCode(@RequestBody TProjectInfo pf, HttpServletResponse response) { // 1.生成源碼文件 // 2.壓縮文件 // 3.設(shè)置回復(fù)的一些參數(shù) // 4.將壓縮文件寫入網(wǎng)絡(luò)流 log.info("request param: {}", pf); OutputStream os = null; try { // 配置文件下載 // 下載文件能正常顯示中文 String filename = pf.getProjectName() + System.currentTimeMillis() + ".zip"; response.setHeader("Content-Disposition", "attachment;filename=" + filename); response.setHeader("Content-Type", "application/octet-stream"); response.setContentType("application/octet-stream; charset=UTF-8"); os = response.getOutputStream(); projectService.generateCode(pf, os); log.info("Download successfully!"); } catch (Exception e) { log.error("Download failed: {}", e.getMessage()); throw new OptErrorException(OptStatus.FAIL.code, "文件下載失敗"); } finally { if (os != null) { try { os.close(); } catch (IOException e) { e.printStackTrace(); } } } }
分析原因
因?yàn)楹蠖酥苯酉騉utputStream寫入,會(huì)覆蓋所有異常捕獲,因此前端直接向data下取字節(jié)流數(shù)據(jù)即可。
正確解法
其實(shí)就是要后端能在錯(cuò)誤時(shí)返回json數(shù)據(jù),正確下載時(shí)直接取data下取字節(jié)流即可,所以使用 ResponseEntity
返回即可。
@PostMapping(value = "/project/code/download") public ResponseEntity<InputStreamResource> downloadCode(@RequestBody TProjectInfo pf) { log.info("request param: {}", pf); String filename = pf.getProjectName() + System.currentTimeMillis() + ".zip"; byte[] bytes = projectService.generateCode(pf); ByteArrayInputStream bais = new ByteArrayInputStream(bytes); HttpHeaders headers = new HttpHeaders(); headers.add("Content-Disposition", String.format("attachment; filename=\"%s\"", filename)); return ResponseEntity.ok() .headers(headers) .contentType(MediaType.parseMediaType("application/octet-stream")) .body(new InputStreamResource(bais)); }
這里的異常,使用RestExceptionAdvice統(tǒng)一處理:
@RestControllerAdvice public class ExceptionAdvice { private static final Logger logger = LoggerFactory.getLogger(ExceptionAdvice.class); /** * 處理數(shù)據(jù)綁定異常 * * @param bindException 數(shù)據(jù)綁定異常 * @return 統(tǒng)一響應(yīng)基本結(jié)果 */ @ExceptionHandler(value = BindException.class) public ResponseEntity<Result<Object>> handleValidateException(BindException bindException) { logger.error("數(shù)據(jù)綁定異常,{}", bindException.getMessage(), bindException); return of(HttpStatus.BAD_REQUEST, "數(shù)據(jù)綁定異常"); } /** * 統(tǒng)一處理 405 異常 * * @param exception 請(qǐng)求方法不支持異常 * @return 統(tǒng)一響應(yīng)結(jié)果 */ @ExceptionHandler(value = HttpRequestMethodNotSupportedException.class) public ResponseEntity<Result<Object>> handleMethodNotSupportedException(HttpRequestMethodNotSupportedException exception) { if (exception != null) { logger.error("Http 405, {}", exception.getMessage(), exception); } return of(HttpStatus.METHOD_NOT_ALLOWED, "方法不被支持"); } private ResponseEntity<Result<Object>> of(HttpStatus status, String msg) { return ResponseEntity.status(status).body(Result.builder().code(OptStatus.FAIL.code).msg(msg).build()); } /** * 統(tǒng)一處理自定義操作錯(cuò)誤異常 * * @param exception 操作錯(cuò)誤異常 * @return 統(tǒng)一響應(yīng)結(jié)果 */ @ExceptionHandler(value = OptErrorException.class) public ResponseEntity<Result<Object>> handleOptErrorException(OptErrorException exception) { return of(HttpStatus.INTERNAL_SERVER_ERROR, exception.getOptMsg()); } /** * 統(tǒng)一處理 服務(wù)器內(nèi)部 異常 * * @param e 異常 * @return 統(tǒng)一響應(yīng)結(jié)果 */ @ExceptionHandler(value = Throwable.class) public ResponseEntity<Result<Object>> handle500(Throwable e) { if (e != null) { logger.error("Http 500, {}", e.getMessage(), e); } return of(HttpStatus.INTERNAL_SERVER_ERROR, "服務(wù)器內(nèi)部未知錯(cuò)誤"); } }
以上就是解決的全過程,可能寫得有點(diǎn)片面,純屬一點(diǎn)個(gè)人實(shí)踐中的見解。
總結(jié)
到此這篇關(guān)于SpringBoot下載文件的正確解法方式的文章就介紹到這了,更多相關(guān)SpringBoot下載文件內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Gradle的SpringBoot項(xiàng)目構(gòu)建圖解
這篇文章主要介紹了Gradle的SpringBoot項(xiàng)目構(gòu)建圖解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-01-01Spring靜態(tài)代理和動(dòng)態(tài)代理代碼詳解
這篇文章主要介紹了Spring靜態(tài)代理和動(dòng)態(tài)代理代碼詳解,具有一定參考價(jià)值,需要的朋友可以了解下。2017-11-11SpringBoot集成內(nèi)存數(shù)據(jù)庫hsqldb的實(shí)踐
hsqldb只需要添加對(duì)應(yīng)的依賴,然后在配置文件進(jìn)行配置。不需要安裝一個(gè)數(shù)據(jù)庫,本文就來介紹一下具體使用,感興趣的可以了解一下2021-09-09JavaWeb項(xiàng)目中springmvc和tomcat對(duì)靜態(tài)文件的處理
這篇文章主要介紹了JavaWeb項(xiàng)目中springmvc和tomcat對(duì)靜態(tài)文件的處理 的相關(guān)資料,需要的朋友可以參考下2016-07-07Springboot?定時(shí)任務(wù)分布式下冪等性解決方案
這篇文章主要介紹了Springboot定時(shí)任務(wù)分布式下冪等性如何解決,本文通過示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-07-07