-
包括:高級參數綁定、@RequestMapping、控制器方法返回值、圖片上傳、Json數據交互、自定義異常處理、自定義攔截器
-
高級參數獲取:
- 數組:
提交表單代碼:
<input type="hidden" name="ids" value="${item.id }" /> <input type="hidden" name="ids" value="${item.id }" /> <input type="hidden" name="ids" value="${item.id }" />
處理器方法:
public ModelAndView function(Integer[] ids){ ****** }
- List集合:
提交表單:
<c:forEach items="${itemList }" var="item" varStatus="v"> <input type="hidden" name="idList[${v.index}]" value="${item.id }" /> <input type="hidden" name="idList[${v.index}]" value="${item.id }" /> <input type="hidden" name="idList[${v.index}]" value="${item.id }" /> </c:forEach>
處理器方法:
public ModelAndView function(QueryVo qv){ ****** }
QueryVo對象中有參數:
private List<Integer> idList;
-
@RequestMapping的使用:
// 方式1: @RequestMapping(value = "/item/itemList.action") public ModelAndView function(QueryVo qv){ ****** } // 方式2: @RequestMapping("/item/itemList.action") public ModelAndView function(QueryVo qv){ ****** }
以上兩種方式,是一樣的,即:當一個方法只設置一個請求路徑,則可以省略" value = "。
那麼當如果一個方法想設置兩個請求路徑。則可以:
@RequestMapping(value = {"/item/itemList.action", "/item/items.action"}) public ModelAndView function(QueryVo qv){ ****** }
這是因爲value屬性的底層是String數組的類型:
public abstract java.lang.String[] value() default {};
@RequestMapping不僅可以使用在方法上,還可以使用在類上,一般情況下,使用在類上時,相當與給類中的方法生命了一個名稱空間,如:
方式一: public class ItemController{ @RequestMapping("/item/itemList.action") public ModelAndView function(QueryVo qv){ ****** } @RequestMapping("/item/selectItem.action") public ModelAndView function(QueryVo qv){ ****** } } 方式二: @RequestMapping("/item") public class ItemController{ @RequestMapping("/itemList.action") public ModelAndView function(QueryVo qv){ ****** } @RequestMapping("/selectItem.action") public ModelAndView function(QueryVo qv){ ****** } }
以上兩種方式起到相同的效果。
@RequestMapping還可以指定的請求方式:
@RequestMapping(method = RequestMethod.POST) @RequestMapping(method = RequestMethod.GET)
指定之後,被指定的控制器方法只能被對應的請求方式請求。
-
控制器方法返回值。
ModelAndView:——無敵方法(推薦少用)
如:
public ModelAndView selectAllItem() { // 創建對象 ModelAndView mav = new ModelAndView(); // 添加返回數據 mav.addObject("hello","HelloWord"); // 指定回顯頁面 mav.setViewName("/jsp/hello.jsp"); return mav; }
String:——此種方式,返回的字符串爲回顯的頁面路徑,並可以設置轉發和重定向。如果想在此種方法中返回數據,需要在聲明一個Model類型的參數。一般用於AJAX異步請求時。如:
public String selectAllItem(Model model) { //設置返回數據 model.addAttribute("hello", "HelloWord"); // 設置回顯頁面 return "/jsp/hello.jsp"; // return "redirect:/jsp/hello.jsp"; // 重定向 // return "forword:/jsp/hello.jsp"; // 轉發 }
Void:——此種方式,相當於原始的Servlet中的方法。沒有特殊的回顯和回執數據方式,只能通過原始的Request和Response方式。
public void queryItem(HttpServletRequest request, HttpServletResponse response) throws Exception { // 1 使用request進行轉發 // request.getRequestDispatcher("/WEB-INF/jsp/success.jsp").forward(request, // response); // 2 使用response進行重定向到編輯頁面 // response.sendRedirect("/springmvc-web2/itemEdit.action"); // 3 使用response直接顯示 response.getWriter().print("{\"abc\":123}"); }
-
圖片上傳:
- 在tomcat上配置圖片虛擬目錄,在tomcat下conf/server.xml中添加:<Context docBase="D:\develop\upload\temp" path="/pic" reloadable="false"/>訪問http://localhost:8080/pic即可訪問D:\develop\upload\temp下的圖片。
- 或者通過eclipse配置。
- 上傳jar包:
- 在springmvc.xml中配置解析器:
<!-- 文件上傳,id必須設置爲multipartResolver --> <bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver"> <!-- 設置文件上傳大小 --> <property name="maxUploadSize" value="5000000" /> </bean>
- 前臺Form表單中設置:
<form action="" method="post" enctype="multipart/form_data"> ****** </form>
- 處理器方法:
// 其中MultipartFile是必須的,用於接收圖片 public String updateItemById(Item item, MultipartFile pictureFile) throws Exception { // 圖片上傳 // 設置圖片名稱,不能重複,可以使用uuid String picName = UUID.randomUUID().toString(); // 獲取文件名 String oriName = pictureFile.getOriginalFilename(); // 獲取圖片後綴 String extName = oriName.substring(oriName.lastIndexOf(".")); // 開始上傳 pictureFile.transferTo(new File("C:/upload/image/" + picName + extName)); // 設置圖片名到商品中 item.setPic(picName + extName); // --------------------------------------------- // 更新商品 this.itemService.updateItemById(item); return "forward:/itemEdit.action"; }
-
Json數據交互:
@RequestBody——簡而言之:將前臺的json數據自動接收並封裝爲JavaBean對象
作用:
@RequestBody註解用於讀取http請求的內容(字符串),通過springmvc提供的HttpMessageConverter接口將讀到的內容(json數據)轉換爲java對象並綁定到Controller方法的參數上。
@ResponseBody——簡而言之:將後臺JavaBean對象自動轉化爲Json數據並傳遞給前臺。
作用:
@ResponseBody註解用於將Controller的方法返回的對象,通過springmvc提供的HttpMessageConverter接口轉換爲指定格式的數據如:json、xml等,通過Response響應給客戶端。
@RequestMapping("testJson") // @ResponseBody public @ResponseBody Item testJson(@RequestBody Item item) { return item; }
-
自定義異常處理:
思路:
系統中異常包括兩類:預期異常和運行時異常RuntimeException,前者通過捕獲異常從而獲取異常信息,後者主要通過規範代碼開發、測試通過手段減少運行時異常的發生。
系統的dao、service、controller出現都通過throws Exception向上拋出,最後由springmvc前端控制器交由異常處理器進行異常處理,
public class CustomHandleException implements HandlerExceptionResolver { @Override public ModelAndView resolveException(HttpServletRequest request, HttpServletResponse response, Object handler, Exception exception) { // 定義異常信息 String msg; // 判斷異常類型 if (exception instanceof MyException) { // 如果是自定義異常,讀取異常信息 msg = exception.getMessage(); } else { // 如果是運行時異常,則取錯誤堆棧,從堆棧中獲取異常信息 Writer out = new StringWriter(); PrintWriter s = new PrintWriter(out); exception.printStackTrace(s); msg = out.toString(); } // 把錯誤信息發給相關人員,郵件,短信等方式 // TODO // 返回錯誤頁面,給用戶友好頁面顯示錯誤信息 ModelAndView modelAndView = new ModelAndView(); modelAndView.addObject("msg", msg); modelAndView.setViewName("error"); return modelAndView; } }
在springmvc.xml中配置:
<!-- 配置全局異常處理器 --> <bean id="customHandleException" class="cn.itcast.ssm.exception.CustomHandleException"/>
-
自定義攔截器:
public class HandlerInterceptor1 implements HandlerInterceptor { // controller執行後且視圖返回後調用此方法 // 這裏可得到執行controller時的異常信息 // 這裏可記錄操作日誌 @Override public void afterCompletion(HttpServletRequest arg0, HttpServletResponse arg1, Object arg2, Exception arg3) throws Exception { System.out.println("HandlerInterceptor1....afterCompletion"); } // controller執行後但未返回視圖前調用此方法 // 這裏可在返回用戶前對模型數據進行加工處理,比如這裏加入公用信息以便頁面顯示 @Override public void postHandle(HttpServletRequest arg0, HttpServletResponse arg1, Object arg2, ModelAndView arg3) throws Exception { System.out.println("HandlerInterceptor1....postHandle"); } // Controller執行前調用此方法 // 返回true表示繼續執行,返回false中止執行 // 這裏可以加入登錄校驗、權限攔截等 @Override public boolean preHandle(HttpServletRequest arg0, HttpServletResponse arg1, Object arg2) throws Exception { System.out.println("HandlerInterceptor1....preHandle"); // 設置爲true,測試使用 return true; } }
配置:
<!-- 配置攔截器 --> <mvc:interceptors> <mvc:interceptor> <!-- 所有的請求都進入攔截器 --> <mvc:mapping path="/**" /> <!-- 配置具體的攔截器 --> <bean class="cn.itcast.ssm.interceptor.HandlerInterceptor1" /> </mvc:interceptor> <mvc:interceptor> <!-- 所有的請求都進入攔截器 --> <mvc:mapping path="/**" /> <!-- 配置具體的攔截器 --> <bean class="cn.itcast.ssm.interceptor.HandlerInterceptor2" /> </mvc:interceptor> </mvc:interceptors>