SpringBoot下載文件的正確解法方式
前言
最近遇到一個奇怪的需求,前端通過post請求下載壓縮文件,同時會傳給后端一些數(shù)據(jù),用于生成壓縮包。此時后端接口就不僅僅是生成壓縮文件流輸出給前端。而必須要有報錯能力與異常處理能力。即如果后端報錯,前端應(yīng)該是下載不了文件流。
比較一般的解法
一般而言,Spring Boot生成文件流供前端下載,會直接將文件流寫入到 HttpServletResponse.getOutputStream(),然而這樣會有一個問題,無論后端如何報錯,前端都能成功下載文件,因?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寫入,會覆蓋所有異常捕獲,因此前端直接向data下取字節(jié)流數(shù)據(jù)即可。
正確解法
其實(shí)就是要后端能在錯誤時返回json數(shù)據(jù),正確下載時直接取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 請求方法不支持異常
* @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)一處理自定義操作錯誤異常
*
* @param exception 操作錯誤異常
* @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)部未知錯誤");
}
}以上就是解決的全過程,可能寫得有點(diǎn)片面,純屬一點(diǎn)個人實(shí)踐中的見解。
總結(jié)
到此這篇關(guān)于SpringBoot下載文件的正確解法方式的文章就介紹到這了,更多相關(guān)SpringBoot下載文件內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Gradle的SpringBoot項(xiàng)目構(gòu)建圖解
這篇文章主要介紹了Gradle的SpringBoot項(xiàng)目構(gòu)建圖解,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-01-01
SpringBoot集成內(nèi)存數(shù)據(jù)庫hsqldb的實(shí)踐
hsqldb只需要添加對應(yīng)的依賴,然后在配置文件進(jìn)行配置。不需要安裝一個數(shù)據(jù)庫,本文就來介紹一下具體使用,感興趣的可以了解一下2021-09-09
JavaWeb項(xiàng)目中springmvc和tomcat對靜態(tài)文件的處理
這篇文章主要介紹了JavaWeb項(xiàng)目中springmvc和tomcat對靜態(tài)文件的處理 的相關(guān)資料,需要的朋友可以參考下2016-07-07
Springboot?定時任務(wù)分布式下冪等性解決方案
這篇文章主要介紹了Springboot定時任務(wù)分布式下冪等性如何解決,本文通過示例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-07-07

