SpringMVC基于注解方式實(shí)現(xiàn)上傳下載
一、文件下載
1-1、servlet原生方式下載
/**
?* 基于servlet api的文件下載
?*/
@RequestMapping("/download")
public String download(HttpServletRequest request,HttpServletResponse response) throws IOException {
? ? // 獲得當(dāng)前項(xiàng)目路徑下的下載文件(真實(shí)開發(fā)中文件名肯定是從數(shù)據(jù)中讀取的)
? ? String realPath =request.getServletContext().getRealPath("/file/20181129204254948.png");
? ? // 根據(jù)文件路徑封裝成了File對(duì)象
? ? File tmpFile=new File(realPath);
? ? // 可以直接根據(jù)File對(duì)象獲得文件名
? ? String fileName = tmpFile.getName();
? ? // 設(shè)置響應(yīng)頭 content-disposition: 就是設(shè)置文件下載的打開方式,默認(rèn)會(huì)在網(wǎng)頁上打開,
? ? // 設(shè)置attachment;filename= 就是為了以下載方式來打開文件
? ? // "UTF-8"設(shè)置如果文件名有中文就不會(huì)亂碼
? ? response.setHeader("content-disposition", "attachment;filename="+ URLEncoder.encode(fileName, "UTF-8"));
? ? // 根據(jù)文件路徑 封裝成文件輸入流
? ? InputStream in = new FileInputStream(realPath);
? ? int len = 0;
? ? // 聲明了一個(gè)1KB的字節(jié) 的緩沖區(qū)
? ? byte[] buffer = new byte[1024];
? ? // 獲取輸出流
? ? OutputStream out = response.getOutputStream();
? ? // 循環(huán)讀取文件,每次讀1KB,避免內(nèi)存溢出
? ? while ((len = in.read(buffer)) > 0) {
? ? ? ? // 往客戶端寫入
? ? ? ? out.write(buffer,0,len);//將緩沖區(qū)的數(shù)據(jù)輸出到客戶端瀏覽器
? ? }
? ? in.close();
? ? return null;
}以上代碼中需要注意的地方我們?cè)O(shè)置了響應(yīng)頭
response.setHeader("content-disposition", "attachment;filename="+ URLEncoder.encode(fileName, "UTF-8"));
其中content-disposition可以讓前端以文件的形式下載,否則就會(huì)直接在瀏覽器打開了
1-2、使用ResponseEntity實(shí)現(xiàn)下載
可以同時(shí)定制響應(yīng)數(shù)據(jù)的內(nèi)容、響應(yīng)頭以及響應(yīng)狀態(tài)碼
1-2-1、使用ResponseEntity實(shí)現(xiàn)響應(yīng)內(nèi)容的定制。
一般在前后端分離的場景中,前端請(qǐng)求后端的接口,后端一般會(huì)返回三個(gè)值,分別為:請(qǐng)求狀態(tài)、請(qǐng)求數(shù)據(jù),以及請(qǐng)求信息。使用ResponseEntity就可以幫助我們定制這樣的內(nèi)容。

通過上圖我們可以看到,通過ResponseEntity可以返回相關(guān)數(shù)據(jù)、也可以設(shè)置響應(yīng)頭、以及狀態(tài)碼。
另外需要注意的是,可以看到方法的返回類型是是Response< String> 那我們?cè)诜祷豏esponseEntity的時(shí)候,第一個(gè)參數(shù)一定是String類型。這個(gè)就是返回泛型的值。
2-2-2、使用ResponseEntity下載文件
/**
?* 基于servlet api的文件下載
?*/
@RequestMapping("/download02")
public ResponseEntity<byte[]> ?download02(HttpServletRequest request) throws IOException {
? ? // 獲得當(dāng)前項(xiàng)目路徑下的下載文件(真實(shí)開發(fā)中文件名肯定是從數(shù)據(jù)中讀取的)
? ? String realPath =request.getServletContext().getRealPath("/file/20181129204254948.png");
? ? // 根據(jù)文件路徑封裝成了File對(duì)象
? ? File tmpFile=new File(realPath);
? ? // 可以直接根據(jù)File對(duì)象獲得文件名
? ? String fileName = tmpFile.getName();
? ? HttpHeaders headers=new HttpHeaders();
? ? headers.set("content-disposition", "attachment;filename="+ URLEncoder.encode(fileName, "UTF-8"));
? ? // 根據(jù)文件路徑 封裝成文件輸入流
? ? InputStream in = new FileInputStream(realPath);
? ? return new ResponseEntity<>(new byte[in.available()],headers,HttpStatus.OK);
}
可以看到使用ResponseEntity也可以同樣下載數(shù)據(jù),但是無法設(shè)置緩沖區(qū),只能全部一次性讀取。
1-2-3、servlet下載和ResponseEntity下載區(qū)別
兩者區(qū)別,使用原生servlet下載,我們可以設(shè)置緩沖區(qū),但是使用ResponseEntity就無法進(jìn)行設(shè)置,只能將文件的全部數(shù)據(jù)以字節(jié)數(shù)組的方式一次性讀取。為了避免內(nèi)存溢出,建議使用servlet原生的方式下載
二、文件上傳
Spring MVC 為文件上傳提供了直接的支持,這種支持是通過即插即用的 MultipartResolver 實(shí)現(xiàn)的。Spring 用 Jakarta Commons FileUpload 技術(shù)實(shí)現(xiàn)了一個(gè) MultipartResolver 實(shí)現(xiàn)類:CommonsMultipartResovler
Spring MVC 上下文中默認(rèn)沒有裝配 MultipartResovler,因此默認(rèn)情況下不能處理文件的上傳工作,如果想使用 Spring 的文件上傳功能,需現(xiàn)在上下文中配置 MultipartResolver。
2-1、添加commons-fileupload依賴
下載基于Jakarta Commons FileUpload的上傳支持jar包
<!-- https://mvnrepository.com/artifact/commons-fileupload/commons-fileupload --> <dependency> ? ? <groupId>commons-fileupload</groupId> ? ? <artifactId>commons-fileupload</artifactId> ? ? <version>1.4</version> </dependency>
如果使用idea一定要手動(dòng)再導(dǎo)入一下包(導(dǎo)入方式可以看之前的文章)

2-2、配置spring.xml注入CommonsMultipartResolver文件上傳解析器
<!--注入基于CommonsMultipartResolver文件上傳解析器-->
<bean class="org.springframework.web.multipart.commons.CommonsMultipartResolver" id="multipartResolver">
<!--設(shè)置編碼集,防止出現(xiàn)中文亂碼-->
<property name="defaultEncoding" value="UTF-8"></property>
<!--設(shè)置最大上傳字節(jié) 配置10MB-->
<property name="maxUploadSize" value="#{1024*1024*10}"></property>
</bean>2-3、文件上傳
2-3-1、單個(gè)文件上傳
2-3-1-1、編寫控制器方法
?
/**
?* 基于springmvc MultiPartResolver文件上傳
?* @param desc
?* @param multipartFile
?* @return
?* @throws IOException
?*/
@PostMapping("/upload01")
public String upload01(String desc, @RequestParam("myfile") MultipartFile multipartFile) throws ?IOException {
? ? System.out.println(desc);
? ? System.out.println(multipartFile.getOriginalFilename());
? ? String path = "d:\\img\\" + multipartFile.getOriginalFilename();
? ? File file = new File(path);
? ? multipartFile.transferTo(file);
? ? return "success";
}
?2-3-1-2、視圖層
<form enctype="multipart/form-data" action="${pageContext.request.contextPath}/upload01" method="post">
<p>文件描述:<input type="text" name="desc" /></p>
<p>文件:<input type="file" name="myfile" accept="image/*"/></p>
<p><input type="submit" value="上傳單個(gè)文件"></p>
</form>2-3-2、多文件上傳
2-3-2-1、編寫控制器方法
/**
?* 基于springmvc MultiPartResolver多文件文件上傳
?* @param desc
?* @param myfile
?* @return
?* @throws IOException
?*/
@PostMapping("/upload02")
public String upload02(String desc,MultipartFile[] myfile) throws ?IOException {
? ? for (MultipartFile multipartFile : myfile) {
? ? ? ? System.out.println(desc);
? ? ? ? System.out.println(multipartFile.getOriginalFilename());
? ? ? ? String path = "d:\img\" + multipartFile.getOriginalFilename();
? ? ? ? File file = new File(path);
? ? ? ? multipartFile.transferTo(file);
? ? }
? ? return "success";
}2-3-2-2、視圖層
視圖層我們可以控制file上傳的文件是否多選可以使用mutiple="mutiple" 在h5中如果屬性和值相等,可以將值省略,然后設(shè)置了accept屬性,可以在用戶選擇的時(shí)候自動(dòng)過濾,如下面代碼只顯示圖片類型的文件
<form enctype="multipart/form-data" action="${pageContext.request.contextPath}/upload03" method="post">
<p>文件描述:<input type="text" name="desc" /></p>
<p>文件:<input type="file" name="myfile" multiple accept="image/*"/></p>
<p><input type="submit" value="上傳多個(gè)文件"></p>
</form>2-3-3、通過多線程的方式批量上傳文件
上面舉例中,使用了多文件上傳,利用for的方式雖然可以逐個(gè)讀取文件并上傳,但是在某些場景下,這樣就效率就降低了很多,為了提高我們上傳的效率,可以利用多線程的方式來進(jìn)行上傳。
2-3-3-4、編寫控制器方法
/**
?* 基于springmvc MultiPartResolver多文件文件上傳--多線程
?* @param desc
?* @param myfile
?* @return
?* @throws IOException
?*/
@PostMapping("/upload03")
public String upload03(String desc,MultipartFile[] myfile) throws IOException, InterruptedException {
? ? System.out.println(desc);
? ? for (MultipartFile multipartFile : myfile) {
? ? ? ? // 聲明線程
? ? ? ? Thread thread = new Thread(() -> {
? ? ? ? ? ? System.out.println(multipartFile.getOriginalFilename());
? ? ? ? ? ? String path = "d:\img\" + multipartFile.getOriginalFilename();
? ? ? ? ? ? File file = new File(path);
? ? ? ? ? ? try {
? ? ? ? ? ? ? ? multipartFile.transferTo(file);
? ? ? ? ? ? } catch (IOException e) {
? ? ? ? ? ? ? ? e.printStackTrace();
? ? ? ? ? ? }
? ? ? ? });
? ? ? ? thread.start(); ? //啟動(dòng)線程
? ? ? ? thread.join(); ? // 讓子線程執(zhí)行完再執(zhí)行主線程
? ? }
? ? return "success";
}到此這篇關(guān)于SpringMVC基于注解方式實(shí)現(xiàn)上傳下載的文章就介紹到這了,更多相關(guān)SpringMVC注解實(shí)現(xiàn)上傳下載內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java中的數(shù)組流ByteArrayOutputStream用法
Java中的ByteArrayOutputStream是java.io包中的一個(gè)類,用于在內(nèi)存中創(chuàng)建字節(jié)數(shù)組緩沖區(qū),支持動(dòng)態(tài)擴(kuò)展,它繼承自O(shè)utputStream,允許以字節(jié)形式寫入數(shù)據(jù),無需與外部設(shè)備交互,常用方法包括write()、toByteArray()、toString()等2024-09-09
SocketIo+SpringMvc實(shí)現(xiàn)文件的上傳下載功能
這篇文章主要介紹了SocketIo+SpringMvc實(shí)現(xiàn)文件的上傳下載功能,socketIo不僅可以用來做聊天工具,也可以實(shí)現(xiàn)局域網(wǎng)。文中給出了實(shí)現(xiàn)代碼,需要的朋友可以參考下2018-08-08
Java調(diào)用騰訊云短信API接口的實(shí)現(xiàn)
這篇文章主要介紹了Java調(diào)用騰訊云短信API接口的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-07-07
線程局部變量的實(shí)現(xiàn)?ThreadLocal使用及場景介紹
這篇文章主要為大家介紹了線程局部變量的實(shí)現(xiàn)?ThreadLocal使用及場景詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-01-01

