SPRing MVC上下文中默認沒有裝配MultipartResesolver,因此默認情況下其不能處理文件上傳工作。如果想使用Spring的文件上傳功能,則需要在springMvc配置文件中配置MultipartResolver
<bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver"> <!-- 上傳文件大小上限,單位為字節(10MB) --> <property name="maxUploadSize"> <value>10485760</value> </property> <!-- 請求的編碼格式,必須和jsp的pageEncoding屬性一致,以便正確讀取表單的內容,默認為ISO-8859-1 --> <property name="defaultEncoding"> <value>UTF-8</value> </property> </bean>jsp:
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%><!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"><html><head><meta http-equiv="Content-Type" content="text/html; charset=UTF-8"><title>文件上傳</title></head><body> <h2>文件上傳</h2> <form action="upload" enctype="multipart/form-data" method="post"> <table> <tr> <td>文件描述:</td> <td><input type="text" name="description"></td> </tr> <tr> <td>請選擇文件:</td> <td><input type="file" name="file"></td> </tr> <tr> <td><input type="submit" value="上傳"></td> </tr> </table> </form></body></html>負責上傳文件的表單的編碼類型必須是“multiart/form-data”.后臺controller代碼:// 上傳文件會自動綁定到MultipartFile中 @RequestMapping(value="/upload",method=RequestMethod.POST) public String upload(HttpServletRequest request, @RequestParam("description") String description, @RequestParam("file") MultipartFile file) throws Exception{ System.out.println(description); // 如果文件不為空,寫入上傳路徑 if(!file.isEmpty()){ // 上傳文件路徑 String path = request.getServletContext().getRealPath( "/images/"); // 上傳文件名 String filename = file.getOriginalFilename(); File filepath = new File(path,filename); // 判斷路徑是否存在,如果不存在就創建一個 if (!filepath.getParentFile().exists()) { filepath.getParentFile().mkdirs(); } // 將上傳文件保存到一個目標文件當中 file.transferTo(new File(path+File.separator+ filename)); return "success"; }else{ return "error"; } }這里一定要把Apache Commons FileUpload的jar包放到項目的類路徑下。要不然會報以下錯誤:
文件下載
模擬的是先上傳后下載:
先上傳圖像后:
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%><!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"><html><head><meta http-equiv="Content-Type" content="text/html; charset=UTF-8"><title>用戶注冊</title></head><body> <h2>用戶注冊</h2> <form action="register" enctype="multipart/form-data" method="post"> <table> <tr> <td>用戶名:</td> <td><input type="text" name="username"></td> </tr> <tr> <td>請上傳頭像:</td> <td><input type="file" name="image"></td> </tr> <tr> <td><input type="submit" value="注冊"></td> </tr> </table> </form></body></html>再后臺下載:<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%><!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"><html><head><meta http-equiv="Content-Type" content="text/html; charset=UTF-8"><title>文件下載</title></head><body><h3>文件下載</h3><a href="download?filename=${requestScope.user.image.originalFilename}">${requestScope.user.image.originalFilename }</a></body></html>后臺代碼:@RequestMapping(value="/register") public String register(HttpServletRequest request, @ModelAttribute User user, Model model)throws Exception{ System.out.println(user.getUsername()); // 如果文件不為空,寫入上傳路徑 if(!user.getImage().isEmpty()){ // 上傳文件路徑 String path = request.getServletContext().getRealPath( "/images/"); // 上傳文件名 String filename = user.getImage().getOriginalFilename(); File filepath = new File(path,filename); // 判斷路徑是否存在,如果不存在就創建一個 if (!filepath.getParentFile().exists()) { filepath.getParentFile().mkdirs(); } // 將上傳文件保存到一個目標文件當中 user.getImage().transferTo(new File(path+File.separator+ filename)); // 將用戶添加到model model.addAttribute("user", user); return "userInfo"; }else{ return "error"; } } @RequestMapping(value="/download") public ResponseEntity<byte[]> download(HttpServletRequest request, @RequestParam("filename") String filename, Model model)throws Exception{ // 下載文件路徑 String path = request.getServletContext().getRealPath( "/images/"); File file = new File(path+File.separator+ filename); HttpHeaders headers = new HttpHeaders(); // 下載顯示的文件名,解決中文名稱亂碼問題 String downloadFielName = new String(filename.getBytes("UTF-8"),"iso-8859-1"); // 通知瀏覽器以attachment(下載方式)打開圖片 headers.setContentDispositionFormData("attachment", downloadFielName); // application/octet-stream : 二進制流數據(最常見的文件下載)。 headers.setContentType(MediaType.APPLICATION_OCTET_STREAM); // 201 HttpStatus.CREATED return new ResponseEntity<byte[]>(FileUtils.readFileToByteArray(file), headers, HttpStatus.CREATED); }攔截器
SpringMVC 中的Interceptor 攔截請求是通過HandlerInterceptor 來實現的。在SpringMVC 中定義一個Interceptor 非常簡單,主要有兩種方式,第一種方式是要定義的Interceptor類要實現了Spring 的HandlerInterceptor 接口,或者是這個類繼承實現了HandlerInterceptor 接口的類,比如Spring 已經提供的實現了HandlerInterceptor 接口的抽象類HandlerInterceptorAdapter ;
HandlerInterceptor 接口中定義了三個方法,我們就是通過這三個方法來對用戶的請求進行攔截處理的。
(1 )preHandle (HttpServletRequest request, HttpServletResponse response, Object handle) 方法,顧名思義,該方法將在請求處理之前進行調用。SpringMVC 中的Interceptor 是鏈式的調用的,在一個應用中或者說是在一個請求中可以同時存在多個Interceptor 。每個Interceptor 的調用會依據它的聲明順序依次執行,而且最先執行的都是Interceptor 中的preHandle 方法,所以可以在這個方法中進行一些前置初始化操作或者是對當前請求的一個預處理,也可以在這個方法中進行一些判斷來決定請求是否要繼續進行下去。該方法的返回值是布爾值Boolean類型的,當它返回為false 時,表示請求結束,后續的Interceptor 和Controller 都不會再執行;當返回值為true 時就會繼續調用下一個Interceptor 的preHandle 方法,如果已經是最后一個Interceptor 的時候就會是調用當前請求的Controller 方法。
(2 )postHandle (HttpServletRequest request, HttpServletResponse response, Object handle, ModelAndView modelAndView) 方法,由preHandle 方法的解釋我們知道這個方法包括后面要說到的afterCompletion 方法都只能是在當前所屬的Interceptor 的preHandle 方法的返回值為true 時才能被調用。postHandle 方法,顧名思義就是在當前請求進行處理之后,也就是Controller 方法調用之后執行,但是它會在DispatcherServlet 進行視圖返回渲染之前被調用,所以我們可以在這個方法中對Controller 處理之后的ModelAndView 對象進行操作。postHandle 方法被調用的方向跟preHandle 是相反的,也就是說先聲明的Interceptor 的postHandle 方法反而會后執行,這和Struts2 里面的Interceptor 的執行過程有點類型。Struts2 里面的Interceptor 的執行過程也是鏈式的,只是在Struts2 里面需要手動調用ActionInvocation 的invoke 方法來觸發對下一個Interceptor 或者是Action 的調用,然后每一個Interceptor 中在invoke 方法調用之前的內容都是按照聲明順序執行的,而invoke 方法之后的內容就是反向的。
(3 )afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handle, Exception ex) 方法,該方法也是需要當前對應的Interceptor 的preHandle 方法的返回值為true 時才會執行。顧名思義,該方法將在整個請求結束之后,也就是在DispatcherServlet 渲染了對應的視圖之后執行。這個方法的主要作用是用于進行資源清理工作的。
在springMVC配置上直接攔截
<mvc:interceptors> <mvc:interceptor> <mvc:mapping path="/upload/*"/> <!-- 匹配url--> <!-- 使用bean定義一個Interceptor,直接定義在mvc:interceptors根下面的Interceptor將攔截所有的請求 --> <bean class="org.fkit.interceptor.AuthorizationInterceptor"/> </mvc:interceptor> <!-- <bean class="org.fkit.interceptor.AuthorizationInterceptor"/> --> </mvc:interceptors>代碼:package org.fkit.interceptor;import javax.servlet.http.HttpServletRequest;import javax.servlet.http.HttpServletResponse;import org.fkit.domain.User;import org.springframework.web.servlet.HandlerInterceptor;import org.springframework.web.servlet.ModelAndView;/** * 攔截器必須實現HandlerInterceptor接口 * */ public class AuthorizationInterceptor implements HandlerInterceptor { // 不攔截"/loginForm"和"/login"請求 private static final String[] IGNORE_URI = {"/loginForm", "/login",}; /** * 該方法將在整個請求完成之后執行, 主要作用是用于清理資源的, * 該方法也只能在當前Interceptor的preHandle方法的返回值為true時才會執行。 */ @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception exception) throws Exception { System.out.println("AuthorizationInterceptor afterCompletion --> "); } /** * 該方法將在Controller的方法調用之后執行, 方法中可以對ModelAndView進行操作 , * 該方法也只能在當前Interceptor的preHandle方法的返回值為true時才會執行。 */ @Override public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView mv) throws Exception { System.out.println("AuthorizationInterceptor postHandle --> "); } /** * preHandle方法是進行處理器攔截用的,該方法將在Controller處理之前進行調用, * 該方法的返回值為true攔截器才會繼續往下執行,該方法的返回值為false的時候整個請求就結束了。 */ @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { System.out.println("AuthorizationInterceptor preHandle --> "); // flag變量用于判斷用戶是否登錄,默認為false boolean flag = false; //獲取請求的路徑進行判斷 String servletPath = request.getServletPath(); // 判斷請求是否需要攔截 for (String s : IGNORE_URI) { if (servletPath.contains(s)) { flag = true; break; } } // 攔截請求 if (!flag){ // 1.獲取session中的用戶 User user = (User) request.getSession().getAttribute("user"); // 2.判斷用戶是否已經登錄 if(user == null){ // 如果用戶沒有登錄,則設置提示信息,跳轉到登錄頁面 System.out.println("AuthorizationInterceptor攔截請求:"); request.setAttribute("message", "請先登錄再訪問網站"); request.getRequestDispatcher("loginForm").forward(request, response); }else{ // 如果用戶已經登錄,則驗證通過,放行 System.out.println("AuthorizationInterceptor放行請求:"); flag = true; } } return flag; }}
新聞熱點
疑難解答