精講RestTemplate第6篇-文件上傳下載與大文件流式下載
- 2020 年 8 月 12 日
- 筆記
本文是精講RestTemplate第6篇,前篇的blog訪問地址如下:
- 精講RestTemplate第1篇-在Spring或非Spring環境下如何使用
- 精講RestTemplate第2篇-多種底層HTTP客戶端類庫的切換
- 精講RestTemplate第3篇-GET請求使用方法詳解
- 精講RestTemplate第4篇-POST請求方法使用詳解
- 精講RestTemplate第5篇-DELETE、PUT等請求方法使用詳解
RestTemplate是HTTP客戶端庫,所以為了使用RestTemplate進行文件上傳和下載,需要我們先編寫服務端的支援文件上傳和下載的程式。請參考我之前寫的一篇文章:SpringBoot實現本地存儲文件上傳及提供HTTP訪問服務 。按照此文完成學習之後,可以獲得
- 一個以訪問服務URI為”/upload」的文件上傳服務端點
- 服務端點上傳文件成功後會返回一個HTTP連接,可以用來下載文件。
下面我們就開始學習使用RestTemplate是HTTP客戶端庫,進行文件的上傳與下載。
一、文件上傳
寫一個單元測試類,來完成RestTemplate文件上傳功能,具體實現細節參考程式碼注釋
@SpringBootTest
class UpDownLoadTests {
@Resource
private RestTemplate restTemplate;
@Test
void testUpload() {
// 文件上傳服務上傳介面
String url = "//localhost:8888/upload";
// 待上傳的文件(存在客戶端本地磁碟)
String filePath = "D:\\data\\local\\splash.png";
// 封裝請求參數
FileSystemResource resource = new FileSystemResource(new File(filePath));
MultiValueMap<String, Object> param = new LinkedMultiValueMap<>();
param.add("uploadFile", resource); //服務端MultipartFile uploadFile
//param.add("param1", "test"); //服務端如果接受額外參數,可以傳遞
// 發送請求並輸出結果
System.out.println("--- 開始上傳文件 ---");
String result = restTemplate.postForObject(url, param, String.class);
System.out.println("--- 訪問地址:" + result);
}
}
輸出結果如下:
--- 開始上傳文件 ---
--- 訪問地址://localhost:8888/2020/08/12/028b38f1-3f9b-4088-9bea-1af8c18cd619.png
文件上傳之後,可以通過上面的訪問地址,在瀏覽器訪問。或者通過RestTemplate客戶端進行下載。
二、文件下載
執行下列程式碼之後,被下載文件url,會被正確的保存到本地磁碟目錄targetPath。
@Test
void testDownLoad() throws IOException {
// 待下載的文件地址
String url = "//localhost:8888/2020/08/12/028b38f1-3f9b-4088-9bea-1af8c18cd619.png";
ResponseEntity<byte[]> rsp = restTemplate.getForEntity(url, byte[].class);
System.out.println("文件下載請求結果狀態碼:" + rsp.getStatusCode());
// 將下載下來的文件內容保存到本地
String targetPath = "D:\\data\\local\\splash-down.png";
Files.write(Paths.get(targetPath), Objects.requireNonNull(rsp.getBody(),
"未獲取到下載文件"));
}
這種下載方法實際上是將下載文件一次性載入到客戶端本地記憶體,然後從記憶體將文件寫入磁碟。這種方式對於小文件的下載還比較適合,如果文件比較大或者文件下載並發量比較大,容易造成記憶體的大量佔用,從而降低應用的運行效率。
三、大文件下載
這種下載方式的區別在於
- 設置了請求頭APPLICATION_OCTET_STREAM,表示以流的形式進行數據載入
- RequestCallback 結合File.copy保證了接收到一部分文件內容,就向磁碟寫入一部分內容。而不是全部載入到記憶體,最後再寫入磁碟文件。
@Test
void testDownLoadBigFile() throws IOException {
// 待下載的文件地址
String url = "//localhost:8888/2020/08/12/028b38f1-3f9b-4088-9bea-1af8c18cd619.png";
// 文件保存的本地路徑
String targetPath = "D:\\data\\local\\splash-down-big.png";
//定義請求頭的接收類型
RequestCallback requestCallback = request -> request.getHeaders()
.setAccept(Arrays.asList(MediaType.APPLICATION_OCTET_STREAM, MediaType.ALL));
//對響應進行流式處理而不是將其全部載入到記憶體中
restTemplate.execute(url, HttpMethod.GET, requestCallback, clientHttpResponse -> {
Files.copy(clientHttpResponse.getBody(), Paths.get(targetPath));
return null;
});
}
歡迎關注我的部落格,裡面有很多精品合集
- 本文轉載註明出處(必須帶連接,不能只轉文字):字母哥部落格。
覺得對您有幫助的話,幫我點贊、分享!您的支援是我不竭的創作動力! 。另外,筆者最近一段時間輸出了如下的精品內容,期待您的關注。
- 《手摸手教你學Spring Boot2.0》
- 《Spring Security-JWT-OAuth2一本通》
- 《實戰前後端分離RBAC許可權管理系統》
- 《實戰SpringCloud微服務從青銅到王者》
- 《VUE深入淺出系列》