Springmvc文件上傳實(shí)現(xiàn)流程解析
SpringMVC 中對(duì)文件上傳做了封裝,我們可以更加方便的實(shí)現(xiàn)文件上傳。從 Spring3.1
開(kāi)始,對(duì)于文件上傳,提供了兩個(gè)處理器:
- CommonsMultipartResolver
- StandardServletMultipartResolver
第一個(gè)處理器兼容性較好,可以兼容 Servlet3.0 之前的版本,但是它依賴了commons-fileupload 這個(gè)第三方工具,所以如果使用這個(gè),一定要添加 commons-fileupload 依賴。
第二個(gè)處理器兼容性較差,它適用于 Servlet3.0之后的版本,它不依賴第三方工具,使用它,可以直接做文件上傳。
CommonsMultipartResolver
添加依賴
<dependency> <groupId>commons-fileupload</groupId> <artifactId>commons-fileupload</artifactId> <version>1.4</version> </dependency>
配置MultipartResolver
<bean class="org.springframework.web.multipart.commons.CommonsMultipartResolver" id="multipartResolver"/>
注意,這個(gè) Bean 一定要有 id,并且 id 必須是 multipartResolver
創(chuàng)建jsp頁(yè)面
<form action="/upload" method="post" enctype="multipart/form-data"> <input type="file" name="file"> <input type="submit" value="上傳"> </form>
注意文件上傳請(qǐng)求是 POST 請(qǐng)求,enctype 一定是 multipart/form-data
開(kāi)發(fā)文件上傳接口
@Controller public class FileUploadController { SimpleDateFormat sdf = new SimpleDateFormat("/yyyy/MM/dd/"); @RequestMapping("/upload") @ResponseBody public String upload(MultipartFile file, HttpServletRequest req) { String format = sdf.format(new Date()); String realPath = req.getServletContext().getRealPath("/img") + format; File folder = new File(realPath); if (!folder.exists()) { folder.mkdirs(); } String oldName = file.getOriginalFilename(); String newName = UUID.randomUUID().toString() + oldName.substring(oldName.lastIndexOf(".")); try { file.transferTo(new File(folder, newName)); String url = req.getScheme() + "://" + req.getServerName() + ":" + req.getServerPort() + "/img" + format + newName; return url; } catch (IOException e) { e.printStackTrace(); } return "failed"; } }
這個(gè)文件上傳方法中,一共做了四件事:
- 解決文件保存路徑,這里是保存在項(xiàng)目運(yùn)行目錄下的 img 目錄下,然后利用日期繼續(xù)寧分類(lèi)
- 處理文件名問(wèn)題,使用 UUID 做新的文件名,用來(lái)代替舊的文件名,可以有效防止文件名沖突
- 保存文件
- 生成文件訪問(wèn)路徑
這里還有一個(gè)小問(wèn)題,在 SpringMVC 中,靜態(tài)資源默認(rèn)都是被自動(dòng)攔截的,無(wú)法訪問(wèn),意味著上傳成功的圖片無(wú)法訪問(wèn),因此,還需要我們?cè)?SpringMVC 的配置文件中,再添加如下配置:
<mvc:resources mapping="/**" location="/"/>
完成之后,就可以訪問(wèn) jsp 頁(yè)面,做文件上傳了。
當(dāng)然,默認(rèn)的配置不一定滿足我們的需求,我們還可以自己手動(dòng)配置文件上傳大小等:
<bean class="org.springframework.web.multipart.commons.CommonsMultipartResolver" id="multipartResolver"> <!--默認(rèn)的編碼--> <property name="defaultEncoding" value="UTF-8"/> <!--上傳的總文件大小--> <property name="maxUploadSize" value="1048576"/> <!--上傳的單個(gè)文件大小--> <property name="maxUploadSizePerFile" value="1048576"/> <!--內(nèi)存中最大的數(shù)據(jù)量,超過(guò)這個(gè)數(shù)據(jù)量,數(shù)據(jù)就要開(kāi)始往硬盤(pán)中寫(xiě)了--> <property name="maxInMemorySize" value="4096"/> <!--臨時(shí)目錄,超過(guò) maxInMemorySize 配置的大小后,數(shù)據(jù)開(kāi)始往臨時(shí)目錄寫(xiě),等全部上傳完成后,再將數(shù)據(jù)合并到正式的文件上傳目錄--> <property name="uploadTempDir" value="file:///E:\\tmp"/> </bean>
StandardServletMultipartResolver
這種文件上傳方式,不需要依賴第三方 jar(主要是不需要添加 commons-fileupload 這個(gè)依賴),但是也不支持 Servlet3.0 之前的版本。
使用 StandardServletMultipartResolver ,那我們首先在 SpringMVC 的配置文件中,配置這個(gè) Bean:
<bean class="org.springframework.web.multipart.support.StandardServletMultipartResolver" id="multipartResolver"></bean>
注意,這里 Bean 的名字依然叫 multipartResolver
配置完成后,注意,這個(gè) Bean 無(wú)法直接配置上傳文件大小等限制。需要在 web.xml 中進(jìn)行配置(這里,即使不需要限制文件上傳大小,也需要在 web.xml 中配置 multipart-config):
<servlet> <servlet-name>springmvc</servlet-name> <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class> <init-param> <param-name>contextConfigLocation</param-name> <param-value>classpath:spring-servlet.xml</param-value> </init-param> <multipart-config> <!--文件保存的臨時(shí)目錄,這個(gè)目錄系統(tǒng)不會(huì)主動(dòng)創(chuàng)建--> <location>E:\\temp</location> <!--上傳的單個(gè)文件大小--> <max-file-size>1048576</max-file-size> <!--上傳的總文件大小--> <max-request-size>1048576</max-request-size> <!--這個(gè)就是內(nèi)存中保存的文件最大大小--> <file-size-threshold>4096</file-size-threshold> </multipart-config> </servlet> <servlet-mapping> <servlet-name>springmvc</servlet-name> <url-pattern>/</url-pattern> </servlet-mapping>
配置完成后,就可以測(cè)試文件上傳了,測(cè)試方式和上面一樣。
多文件上傳
多文件上傳分為兩種,一種是 key 相同的文件,另一種是 key 不同的文件。
1 key 相同的文件
這種上傳,前端頁(yè)面一般如下:
<form action="/upload2" method="post" enctype="multipart/form-data"> <input type="file" name="files" multiple> <input type="submit" value="上傳"> </form>
主要是 input 節(jié)點(diǎn)中多了 multiple 屬性。后端用一個(gè)數(shù)組來(lái)接收文件即可:
@RequestMapping("/upload2") @ResponseBody public void upload2(MultipartFile[] files, HttpServletRequest req) { String format = sdf.format(new Date()); String realPath = req.getServletContext().getRealPath("/img") + format; File folder = new File(realPath); if (!folder.exists()) { folder.mkdirs(); } try { for (MultipartFile file : files) { String oldName = file.getOriginalFilename(); String newName = UUID.randomUUID().toString() + oldName.substring(oldName.lastIndexOf(".")); file.transferTo(new File(folder, newName)); String url = req.getScheme() + "://" + req.getServerName() + ":" + req.getServerPort() + "/img" + format + newName; System.out.println(url); } } catch (IOException e) { e.printStackTrace(); } }
2 key 不同的文件
key 不同的,一般前端定義如下:
<form action="/upload3" method="post" enctype="multipart/form-data"> <input type="file" name="file1"> <input type="file" name="file2"> <input type="submit" value="上傳"> </form>
這種,在后端用不同的變量來(lái)接收就行了:
@RequestMapping("/upload3") @ResponseBody public void upload3(MultipartFile file1, MultipartFile file2, HttpServletRequest req) { String format = sdf.format(new Date()); String realPath = req.getServletContext().getRealPath("/img") + format; File folder = new File(realPath); if (!folder.exists()) { folder.mkdirs(); } try { String oldName = file1.getOriginalFilename(); String newName = UUID.randomUUID().toString() + oldName.substring(oldName.lastIndexOf(".")); file1.transferTo(new File(folder, newName)); String url1 = req.getScheme() + "://" + req.getServerName() + ":" + req.getServerPort() + "/img" + format + newName; System.out.println(url1); String oldName2 = file2.getOriginalFilename(); String newName2 = UUID.randomUUID().toString() + oldName2.substring(oldName2.lastIndexOf(".")); file2.transferTo(new File(folder, newName2)); String url2 = req.getScheme() + "://" + req.getServerName() + ":" + req.getServerPort() + "/img" + format + newName2; System.out.println(url2); } catch (IOException e) { e.printStackTrace(); } }
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Mybatis空值關(guān)聯(lián)的問(wèn)題解析及解決方案
這篇文章給大家介紹了Mybatis空值關(guān)聯(lián)的問(wèn)題解析及解決方案,文中通過(guò)代碼示例介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下2024-01-01java實(shí)現(xiàn)輸出字符串中第一個(gè)出現(xiàn)不重復(fù)的字符詳解
這篇文章主要介紹了java實(shí)現(xiàn)輸出字符串中第一個(gè)出現(xiàn)不重復(fù)的字符詳解的相關(guān)資料,需要的朋友可以參考下2017-04-04淺談Spring 解決循環(huán)依賴必須要三級(jí)緩存嗎
這篇文章主要介紹了淺談Spring 解決循環(huán)依賴必須要三級(jí)緩存嗎,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-10-10IDEA進(jìn)程已結(jié)束,退出代碼-1073741819 (0xC0000005)的bug
這篇文章主要介紹了IDEA進(jìn)程已結(jié)束,退出代碼-1073741819 (0xC0000005)的bug,本文通過(guò)實(shí)例代碼圖文的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-04-04