SpringMVC介紹
Spring web mvc 和Struts2都屬于表現(xiàn)層的框架,它是Spring框架的一部分
Spring
SpringMVC處理流程
處理流程
入門程序
創(chuàng)建工程
入門工程
導(dǎo)入jar包
jar包
在web.xml中配置前端控制器DispatcherServlet
<?xml version="1.0"encoding="UTF-8"?>springmvcorg.springframework.web.servlet.DispatcherServletspringmvc*.action
在上面的url-pattern中攔截規(guī)則有如下:
/* 表示攔截所有 jsp js png .css 會(huì)全部攔截
*.action *.do 攔截以do action 結(jié)尾的請(qǐng)求
/ 攔截所有(不包括jsp) 包含 .js .png .css
創(chuàng)建springmvc.xml
<?xml version="1.0"encoding="UTF-8"?><!--配置controller掃描包-->
創(chuàng)建pojo
publicclassItem{// 商品idprivateintid;// 商品名稱privateString name;// 商品價(jià)格privatedoubleprice;// 商品創(chuàng)建時(shí)間privateDate createtime;// 商品描述privateString detail;創(chuàng)建帶參數(shù)的構(gòu)造器set/get。。。}
創(chuàng)建ItemController
ItemController是一個(gè)普通的java類,不需要實(shí)現(xiàn)任何接口
需要再類上添加@Controller注解,將Controller交由Spring管理
在方法上面添加@RequestMapping注解 里面指定請(qǐng)求的url
@RequestMapping("/itemList.action")publicModelAndViewqueryItemList(){// 創(chuàng)建頁面需要顯示的商品數(shù)據(jù)// 創(chuàng)建頁面需要顯示的商品數(shù)據(jù)Listlist=newArrayList<>();list.add(newItems(1,"1華為 榮耀8",2399f,newDate(),"質(zhì)量好!1"));list.add(newItems(2,"2華為 榮耀8",2399f,newDate(),"質(zhì)量好!2"));list.add(newItems(3,"3華為 榮耀8",2399f,newDate(),"質(zhì)量好!3"));list.add(newItems(4,"4華為 榮耀8",2399f,newDate(),"質(zhì)量好!4"));list.add(newItems(5,"5華為 榮耀8",2399f,newDate(),"質(zhì)量好!5"));list.add(newItems(6,"6華為 榮耀8",2399f,newDate(),"質(zhì)量好!6"));//創(chuàng)建modelandview 模型視圖對(duì)象ModelAndView mav =newModelAndView();//設(shè)置視圖跳轉(zhuǎn)名稱mav.setViewName("WEB-INF/jsp/itemList.jsp");//設(shè)置model數(shù)據(jù)mav.addObject("itemList",list);returnmav;? ? }
啟動(dòng)測(cè)試
測(cè)試
SpringMVC架構(gòu)
框架結(jié)構(gòu)
框架結(jié)構(gòu)
組件說明
以下組件通常由SpringMVC提供實(shí)現(xiàn)
前端控制器 DispatcherServlet
用戶請(qǐng)求到達(dá)前端控制器,它就相當(dāng)于MVC模式中的C,dispatcherServlet是整個(gè)流程控制的中心,由它調(diào)用其它組件處理用戶的請(qǐng)求,DispathcerServlet的存在降低了各個(gè)組件之間的耦合性
處理器映射器 HandlerMapping
HandlerMapping負(fù)責(zé)根據(jù)用戶請(qǐng)求url找到Handler(處理器)springmvc提供了不同的映射器實(shí)現(xiàn)不同的映射方式。
Handler處理器
Handler是繼DispathcerServlet前端控制器的后端控制器,在DispathcerServlet的控制下Handler對(duì)具體的用戶請(qǐng)求進(jìn)行處理由于Handler涉及到具體的用戶業(yè)務(wù)請(qǐng)求,所以一般情況需要自己開發(fā)Handler
HandlerAdapter 處理器適配器
通過HandlerAdapter對(duì)處理器進(jìn)行執(zhí)行,這是適配器模式的應(yīng)用,通過擴(kuò)展適配器可以對(duì)更對(duì)類型的處理器進(jìn)行執(zhí)行
ViewResolver 視圖解析器
ViewResolver 負(fù)責(zé)將處理結(jié)果生成View視圖,ViewResolver首先根據(jù)邏輯視圖名解析成物理視圖名即具體的頁面地址,再生成View視圖對(duì)象,最后對(duì)View進(jìn)行渲染將處理結(jié)果通過頁面展示給客戶
View視圖
springmvc框架提供了很多View視圖類型的支持,包括:jstlView、freemarkerView、pdfView等
說明
在SpringMVC的各個(gè)組件中,處理器映射器,處理器適配器、視圖解析器稱為SpringMVC的三大組件
需要用戶開發(fā)的組件是:Handler View
注解映射器和適配器
配置處理器映射器
注解式處理器映射器,對(duì)類中標(biāo)記了@ResquestMapping的方法進(jìn)行映射。根據(jù)@RequestMapping定義的url匹配@ResquestMapping標(biāo)記的方法,匹配成功返回HandlerMethod對(duì)象給前端控制器。
HandlerMethod對(duì)象中封裝URL對(duì)應(yīng)的方法Method
從Spring3.1開始,廢除了DefaultAnnotationHandlerMapping的使用 拓建使用RequestMappingHandlerMapping完成注解式處理器映射
在springmvc.xml配置文件中配置如下
<!--配置處理器映射器-->
配置處理器適配器
注解式處理器適配器,對(duì)標(biāo)記@RequestMapping的方法進(jìn)行適配
從Spring3.1版本開始,廢除了AnnotationMethodHandlerAdapter的使用,推薦使用RequestMappingHandlerAdapter完成注解式適配器適配
在springmvc.xml配置文件中配置如下
<!--配置處理器適配器-->
注解驅(qū)動(dòng)
直接配置處理器映射器和處理器適配器比較麻煩,可以使用注解驅(qū)動(dòng)來加載 SpringMVC使用 自動(dòng)加載RequestMappingHandlerMapping和RequestMappingHandlerAdapter
可以在springmvc.xml配置文件中使用替代注解處理器和適配器的設(shè)置
視圖解析器
視圖解析器使用SpringMVC框架默認(rèn)的InternalResourceViewResolver,這個(gè)視圖解析器支持JSP視圖解析
在springmvc.xml配置文件中配置如下
<!--視圖解析器--><!--配置邏輯視圖前綴--><!--配置邏輯視圖后綴-->
修改ItemController
// @RequestMapping:里面放的是請(qǐng)求的url,和用戶請(qǐng)求的url進(jìn)行匹配// action可以寫也可以不寫@RequestMapping("/itemList.action")publicModelAndViewqueryItemList(){// 創(chuàng)建頁面需要顯示的商品數(shù)據(jù)Listlist=newArrayList<>();list.add(newItem(1,"1華為 榮耀8",2399,newDate(),"質(zhì)量好!1"));list.add(newItem(2,"2華為 榮耀8",2399,newDate(),"質(zhì)量好!2"));list.add(newItem(3,"3華為 榮耀8",2399,newDate(),"質(zhì)量好!3"));list.add(newItem(4,"4華為 榮耀8",2399,newDate(),"質(zhì)量好!4"));list.add(newItem(5,"5華為 榮耀8",2399,newDate(),"質(zhì)量好!5"));list.add(newItem(6,"6華為 榮耀8",2399,newDate(),"質(zhì)量好!6"));// 創(chuàng)建ModelAndView,用來存放數(shù)據(jù)和視圖ModelAndView modelAndView =newModelAndView();// 設(shè)置數(shù)據(jù)到模型中modelAndView.addObject("itemList",list);// 設(shè)置視圖jsp,需要設(shè)置視圖的物理地址// modelAndView.setViewName("/WEB-INF/jsp/itemList.jsp");// 配置好視圖解析器前綴和后綴,這里只需要設(shè)置邏輯視圖就可以了。// 視圖解析器根據(jù)前綴+邏輯視圖名+后綴拼接出來物理路徑modelAndView.setViewName("itemList");returnmodelAndView;}
整合mybatis
整合目標(biāo):控制層采用SpringMVC 持久層采用mybatis
導(dǎo)入jar包
導(dǎo)入jara包
創(chuàng)建數(shù)據(jù)庫配置文件db.properties
db.properties
jdbc.driver=com.mysql.jdbc.Driverjdbc.url=jdbc:mysql://140.210.70.23:3306/springmvc?characterEncoding=utf-8jdbc.username=appUserjdbc.password=AA1122BB
web.xml
在web.xml加入Spring監(jiān)聽器 ContextLoaderListener 并指定Spring配置文件
web.xml文件中配置SpringMVC的前端控制器
<?xml version="1.0"encoding="UTF-8"?><!--指定配置文件-->contextConfigLocationclasspath:applicationContext.xmlorg.springframework.web.context.ContextLoaderListener<!--spring mvc 前端控制器-->springmvcorg.springframework.web.servlet.DispatcherServlet<!--指定SpringMVC配置文件-->contextConfigLocationclasspath:springmvc.xmlspringmvc*.action
springmvc.xml
在springmvc中配置controller掃描包
視圖解析器
注解驅(qū)動(dòng)
<?xml version="1.0"encoding="UTF-8"?>
? ? ? ? http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-4.0.xsd
? ? ? ? http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.0.xsd"><!--配置controller掃描包--><!--配置處理器映射器--><!--<bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping"/>--><!--配置處理器適配器--><!--<bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter"/>--><!--注解驅(qū)動(dòng)--><!--視圖解析器--><!--配置邏輯視圖前綴--><!--配置邏輯視圖后綴-->
applicationContext.xml
spring配置文件中配置 數(shù)據(jù)庫文件,連接池
配置mybatis的工廠并指定核心配置文件的位置
指定mapper動(dòng)態(tài)代理開發(fā)的掃描
<?xml version="1.0"encoding="UTF-8"?>
? ? http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.0.xsd
? ? ? http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-4.0.xsd"><!-- 數(shù)據(jù)庫連接池 --><!-- Mybatis的工廠 --><!-- 核心配置文件的位置 --><!-- Mapper動(dòng)態(tài)代理開發(fā)? 掃描 --><!-- 基本包 -->
參數(shù)綁定
在開發(fā)中 需要從請(qǐng)求的參數(shù)中把請(qǐng)求的id取出來
id包含在Request對(duì)象中,可以從Request對(duì)象中取id
想獲得Request對(duì)象需要再Controller方法的形參中添加一個(gè)參數(shù)即可。Springmvc框架會(huì)自動(dòng)把Request對(duì)象傳遞給方法
jsp頁面
修改
Controller
/**? ? * 通過id查詢商品? ? *? ? *@paramrequest? ? *@return*/@RequestMapping("/itemEdit")publicModelAndViewqueryItemById(HttpServletRequest request){? ? ? ? String strId = request.getParameter("id");? ? ? ? Items items = itemService.queryItemById(Integer.valueOf(strId));//傳遞結(jié)果回頁面ModelAndView mva =newModelAndView();? ? ? ? mva.setViewName("editItem");//設(shè)置數(shù)據(jù)mva.addObject("item", items);returnmva;? ? }
默認(rèn)支持的參數(shù)類型
處理器形參中添加如下類型的參數(shù)處理適配器會(huì)默認(rèn)識(shí)別并進(jìn)行賦值
HttpServletRequest 通過request對(duì)象獲取請(qǐng)求信息
HttpServletResponse 通過response處理響應(yīng)信息
HttpSession 通過session對(duì)象得到session中存放的對(duì)象
Model/ModelMap
Model
除了ModelAndView之外,還可以使用Model來向頁面?zhèn)鬟f數(shù)據(jù)
Model是一個(gè)接口,在參數(shù)里直接聲明model即可
如果使用Model則可以不適用ModelAndView對(duì)象,Model對(duì)象可以向頁面?zhèn)鬟f數(shù)據(jù)View對(duì)象則可以使用String返回值替代
不管是Model還是ModelAndView其本質(zhì)都是使用Request對(duì)象向jsp傳遞數(shù)據(jù)
/** * 根據(jù)id查詢商品,使用Model *? *@paramrequest *@parammodel *@return*/@RequestMapping("/itemEdit")publicStringqueryItemById(HttpServletRequest request, Model model){// 從request中獲取請(qǐng)求參數(shù)String strId = request.getParameter("id");? ? Integer id = Integer.valueOf(strId);// 根據(jù)id查詢商品數(shù)據(jù)Item item =this.itemService.queryItemById(id);// 把結(jié)果傳遞給頁面// ModelAndView modelAndView = new ModelAndView();// 把商品數(shù)據(jù)放在模型中// modelAndView.addObject("item", item);// 設(shè)置邏輯視圖// modelAndView.setViewName("itemEdit");// 把商品數(shù)據(jù)放在模型中model.addAttribute("item", item);return"itemEdit";}
ModelMap
ModelMap是Model接口的實(shí)現(xiàn)類,也可以通過ModelMap向頁面?zhèn)鬟f數(shù)據(jù)。
使用Model和ModelMap的效果一樣,如果直接使用Model,springmvc會(huì)實(shí)例化ModelMap
*@paramrequest *@parammodel *@return*/@RequestMapping("/itemEdit")publicStringqueryItemById(HttpServletRequest request, ModelMap model){// 從request中獲取請(qǐng)求參數(shù)String strId = request.getParameter("id");? ? Integer id = Integer.valueOf(strId);// 根據(jù)id查詢商品數(shù)據(jù)Item item =this.itemService.queryItemById(id);// 把結(jié)果傳遞給頁面// ModelAndView modelAndView = new ModelAndView();// 把商品數(shù)據(jù)放在模型中// modelAndView.addObject("item", item);// 設(shè)置邏輯視圖// modelAndView.setViewName("itemEdit");// 把商品數(shù)據(jù)放在模型中model.addAttribute("item", item);return"itemEdit";}
綁定簡(jiǎn)單類型
當(dāng)請(qǐng)求的參數(shù)名稱和處理器形參名稱一致時(shí)會(huì)將請(qǐng)求參數(shù)與形參進(jìn)行綁定。這樣從Request取參數(shù)的方法就可以進(jìn)一步簡(jiǎn)化
/** * 根據(jù)id查詢商品,綁定簡(jiǎn)單數(shù)據(jù)類型 *? *@paramid *@parammodel *@return*/@RequestMapping("/itemEdit")publicStringqueryItemById(intid, ModelMap model){// 根據(jù)id查詢商品數(shù)據(jù)Item item =this.itemService.queryItemById(id);// 把商品數(shù)據(jù)放在模型中model.addAttribute("item", item);return"itemEdit";}
支持綁定的數(shù)據(jù)類型
參數(shù)類型推薦使用包裝數(shù)據(jù)類型,因?yàn)榛A(chǔ)數(shù)據(jù)類型不可以為null
整形:Integer、int
字符串:String
單精度:Float、float
雙精度:Double、double
布爾型:Boolean、boolean
說明:對(duì)于布爾類型的參數(shù),請(qǐng)求的參數(shù)值為true或false。或者1或0
請(qǐng)求url:
http://localhost:8080/xxx.action?id=2&status=false
處理器方法:
public String editItem(Model model,Integer id,Boolean status)
@RequestParam
使用@RequestParam常用于處理簡(jiǎn)單類型的綁定
value 參數(shù)名字,即入?yún)⒌恼?qǐng)求參數(shù)名字
required 是否必須 默認(rèn)是true 表示請(qǐng)求中一定要有響應(yīng)的參數(shù) 否則報(bào)錯(cuò)
default 默認(rèn)值,表示如果請(qǐng)求中沒有同名參數(shù)時(shí)的默認(rèn)值
@RequestMapping("/itemEdit")publicStringqueryItemById(@RequestParam(value ="itemId", required =true, defaultValue ="1")Integer id,? ? ? ? ModelMap modelMap){// 根據(jù)id查詢商品數(shù)據(jù)Item item =this.itemService.queryItemById(id);// 把商品數(shù)據(jù)放在模型中modelMap.addAttribute("item", item);return"itemEdit";}
綁定POJO類型
如果提交的參數(shù)很多,或者提交的表單中的內(nèi)容很多的時(shí)候,可以使用簡(jiǎn)單類型接受數(shù)據(jù),也可以使用pojo接收數(shù)據(jù)。
要求:pojo對(duì)象中的屬性名和表單中的input的name屬性一致
Controller
@RequestMapping(value ="/updateitem.action")publicStringupdateitem(Items items){? ? ? ? itemService.updateItemsById(items);return"success";? ? }
處理亂碼問題
加入過濾器
<!--編碼過濾器-->encodingorg.springframework.web.filter.CharacterEncodingFilterencodingUTF-8encoding*.action
綁定包裝POJO
創(chuàng)建包裝的POJO QueryVo
packagecn.probuing.springmvc.pojo;/** *@Auther: wxblack-mac *@Date: 2018/8/28 17:45 *@Description: */publicclassQueryVo{privateItems items;publicItemsgetItems(){returnitems;? ? }publicvoidsetItems(Items items){this.items = items;? ? }}
jsp頁面
<!-- <form id="itemForm" action="" method="post" enctype="multipart/form-data"> -->修改商品信息:商品名稱商品價(jià)格
自定義參數(shù)綁定
有時(shí)候參數(shù)無法被springmvc轉(zhuǎn)換,我們需要自定義轉(zhuǎn)換參數(shù)
自定義Converter 實(shí)現(xiàn)Converter接口 泛型為 源類型和轉(zhuǎn)換后的類型
/** *@Auther: wxblack-mac *@Date: 2018/8/28 19:05 * DataConverter * S:需要轉(zhuǎn)換的源的類型 * T:需要轉(zhuǎn)換的目標(biāo)類型 */publicclassDataConverterimplementsConverter{@OverridepublicDateconvert(String s){//轉(zhuǎn)換stry{? ? ? ? ? ? SimpleDateFormat sdf =newSimpleDateFormat("yyy-MM-dd HH:mm:ss");? ? ? ? ? ? Date date = sdf.parse(s);returndate;? ? ? ? }catch(ParseException e) {returnnull;? ? ? ? }? ? }}
配置轉(zhuǎn)換器Converter
在springmvc.xml文件中
<!--配置轉(zhuǎn)換器工廠bean--><!--配置轉(zhuǎn)換器--><!--指定自定義轉(zhuǎn)換器實(shí)現(xiàn)類-->
高級(jí)參數(shù)綁定
數(shù)組參數(shù)
@RequestMapping(value ="deletes.action")publicStringdeletes(Integer[] ids){for(Integer id : ids) {? ? ? ? ? ? System.out.println(id+"111");? ? ? ? }return"success";? ? }
集合參數(shù)
"/>修改
SpringMVC和Struts2區(qū)別
1、SpringMVC的入口是一個(gè)Servlet即前端控制器。而Struts2入口是一個(gè)filter過濾器
2、SpringMVC請(qǐng)求參數(shù)傳遞到方法的形參,Struts2傳遞參數(shù)是通過類的屬性
3、SpringMVC是基于方法開發(fā)(一個(gè)url對(duì)應(yīng)一個(gè)方法),可以設(shè)計(jì)成單例或者多例
4、Struts采用值棧存儲(chǔ)請(qǐng)求和響應(yīng)數(shù)據(jù),通過OGNL存儲(chǔ)數(shù)據(jù) SpringMVC通過參數(shù)計(jì)息期將request請(qǐng)求內(nèi)容解析,并給方法形參賦值,將數(shù)據(jù)和視圖封裝成ModelAndView對(duì)象,最后又將ModelAndView中的模型數(shù)據(jù)通過request域傳輸?shù)巾撁妫琷sp視圖解析器默認(rèn)使用jstl
@RequestMapping
通過@RequestMapping注解可以定義不同的處理器映射規(guī)則
URL路徑映射
@RequestMapping(value="item")或者 @RequestMapping("/item")
value是一個(gè)數(shù)組,可以將多個(gè)url映射到同一個(gè)方法
/**
* 查詢商品列表
* @return
*/@RequestMapping(value = {"itemList","itemListAll"})publicModelAndViewqueryItemList(){// 查詢商品數(shù)據(jù)Listlist=this.itemService.queryItemList();// 創(chuàng)建ModelAndView,設(shè)置邏輯視圖名ModelAndView mv =newModelAndView("itemList");// 把商品數(shù)據(jù)放到模型中mv.addObject("itemList",list);returnmv;}
添加到類上 窄化路徑
對(duì)于重復(fù)的父路徑可以使用URL放在類上可以指定簡(jiǎn)化的類路徑
//@RequestMapping("item")publicclassItemController{
請(qǐng)求方法限定
method={RequestMethod.POST,RequestMethod.GET} 指定請(qǐng)求方式
限定GET方法
@RequestMapping(method= RequestMethod.GET)
限定POST方法
@RequestMapping(method= RequestMethod.POST)
Controller方法返回值
返回ModelAndView
controller 方法中定義ModelAndView對(duì)象并返回,對(duì)象中可添加moedel數(shù)據(jù),指定View
返回void
在Controller方法形參上可以定義request和response
適合json 和 ajax請(qǐng)求時(shí)的方式
/** * 返回void測(cè)試 *? *@paramrequest *@paramresponse *@throwsException */@RequestMapping("queryItem")publicvoidqueryItem(HttpServletRequest request, HttpServletResponse response)throwsException{// 1 使用request進(jìn)行轉(zhuǎn)發(fā)// request.getRequestDispatcher("/WEB-INF/jsp/success.jsp").forward(request,// response);// 2 使用response進(jìn)行重定向到編輯頁面// response.sendRedirect("/springmvc-web2/itemEdit.action");// 3 使用response直接顯示response.getWriter().print("{\"abc\":123}");}
返回字符串
邏輯視圖名
controller 方法返回字符串可以指定邏輯視圖名,通過視圖解析為物理視圖地址
Redirect重定向
Controller方法返回字符串可以重定向到一個(gè)url地址
/** * 更新商品 *? *@paramitem *@return*/@RequestMapping("updateItem")publicStringupdateItemById(Item item){// 更新商品this.itemService.updateItemById(item);// 修改商品成功后,重定向到商品編輯頁面// 重定向后瀏覽器地址欄變更為重定向的地址,// 重定向相當(dāng)于執(zhí)行了新的request和response,所以之前的請(qǐng)求參數(shù)都會(huì)丟失// 如果要指定請(qǐng)求參數(shù),需要在重定向的url后面添加 ?itemId=1 這樣的請(qǐng)求參數(shù)return"redirect:/itemEdit.action?itemId="+ item.getId();}
forward轉(zhuǎn)發(fā)
Controllrer方法執(zhí)行后繼續(xù)執(zhí)行另一個(gè)Controller方法
/** * 更新商品 *? *@paramitem *@return*/@RequestMapping("updateItem")publicStringupdateItemById(Item item){// 更新商品this.itemService.updateItemById(item);// 修改商品成功后,重定向到商品編輯頁面// 重定向后瀏覽器地址欄變更為重定向的地址,// 重定向相當(dāng)于執(zhí)行了新的request和response,所以之前的請(qǐng)求參數(shù)都會(huì)丟失// 如果要指定請(qǐng)求參數(shù),需要在重定向的url后面添加 ?itemId=1 這樣的請(qǐng)求參數(shù)// return "redirect:/itemEdit.action?itemId=" + item.getId();// 修改商品成功后,繼續(xù)執(zhí)行另一個(gè)方法// 使用轉(zhuǎn)發(fā)的方式實(shí)現(xiàn)。轉(zhuǎn)發(fā)后瀏覽器地址欄還是原來的請(qǐng)求地址,// 轉(zhuǎn)發(fā)并沒有執(zhí)行新的request和response,所以之前的請(qǐng)求參數(shù)都存在return"forward:/itemEdit.action";}
異常處理器
Springmvc在處理請(qǐng)求過程中出現(xiàn)異常信息交由異常處理器進(jìn)行處理,自定義異常處理器可以實(shí)現(xiàn)一個(gè)系統(tǒng)的異常處理邏輯
異常處理思路
系統(tǒng)中異常包括兩類:預(yù)期異常和運(yùn)行時(shí)異常,前者通過捕獲異常從而獲取異常信息,后者主要通過規(guī)范代碼開發(fā)、測(cè)試通過手段減少運(yùn)行時(shí)異常發(fā)生
異常處理過程
異常處理器自定義實(shí)現(xiàn)類
實(shí)現(xiàn)HandlerExceptionResolver接口
/** *@Auther: wxblack-mac *@Date: 2018/8/30 14:50 *@Description: */publicclassMyExceptionHandlerimplementsHandlerExceptionResolver{@OverridepublicModelAndViewresolveException(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, Exception e){? ? ? ? System.out.println(o);? ? ? ? ModelAndView mav =newModelAndView();? ? ? ? mav.addObject("error","未知異常");? ? ? ? mav.setViewName("error");returnmav;? ? }}
方法形參中:httpServletRequest 請(qǐng)求對(duì)象 httpServletResponse 響應(yīng)對(duì)象
Object o:異常信息封裝 包名+類名+方法名
Exception e:異常對(duì)象
返回的ModelAndView 可以指定視圖和數(shù)據(jù)
配置異常處理
<!--自定義異常處理-->
上傳圖片
加入上傳圖片的jar包
jar包
配置文件上傳解析器
文件上傳解析器 id必須設(shè)置為multipartResolver
<!--配置文件文件上傳解析器--><!--設(shè)置文件大小-->
jsp頁面
{item.id }"/> 修改商品信息:
商品名稱
{item.price }"/>
商品生產(chǎn)日期
value=""/>
商品圖片
<%--
--%>
商品簡(jiǎn)介
${item.detail }
</form>
java代碼
方法接收時(shí),圖片形參的名字必須要和jsp頁面 傳遞的圖片的name一致
@RequestMapping(value ="/updateitem.action")publicStringupdateitem(QueryVo vo, MultipartFile pictureFile)throwsIOException{//創(chuàng)建圖片名稱String picName = UUID.randomUUID().toString().replaceAll("-","");//獲取文件名String oriName = pictureFile.getOriginalFilename();//獲取圖片后綴String ext = FilenameUtils.getExtension(oriName);//保存圖片pictureFile.transferTo(newFile("/Users/wxblack-mac/pic/"+ picName +"."+ ext));//保存數(shù)據(jù)庫vo.getItems().setPic(picName +"."+ ext);? ? ? ? itemService.updateItemsById(vo.getItems());return"redirect:/itemEdit.action?id="+ vo.getItems().getId();? ? }
JSON數(shù)據(jù)交互
@RequestBody
@RequestBody注解用于讀取http請(qǐng)求的內(nèi)容,通過springmvc提供的HttpMessageConverter接口將讀取到的內(nèi)容轉(zhuǎn)換為java對(duì)象并綁定到Controller方法的參數(shù)上
@ResponseBody
@ReponseBody注解用于將Controller的方法返回的對(duì)象,通過SpringMVC提供的HttpMessageConverter接口轉(zhuǎn)換為指定格式的數(shù)據(jù)。通過Response響應(yīng)到客戶端
加入jar包
圖片.png
Controller
@RequestMapping(value ="/json.action")public@ResponseBodyItemsjson(@RequestBody Items items){returnitems;? ? }
JS發(fā)送Json數(shù)據(jù)
$(function(){varparams ='{"id": 1,"name": "測(cè)試商品","price": 99.9,"detail": "測(cè)試商品描述","pic": "123456.jpg"}';? ? ? ? ? ? alert(params);? ? ? ? ? ? $.ajax({url:"${pageContext.request.contextPath}/json.action",data: params,contentType:"application/json;charset=UTF-8",type:"post",dataType:"json",success:function(data){? ? ? ? ? ? ? ? ? ? alert(data.name);? ? ? ? ? ? ? ? }? ? ? ? ? ? });? ? ? ? });
配置Json轉(zhuǎn)換器
如果不適用注解驅(qū)動(dòng),就需要給處理器適配器配置json轉(zhuǎn)換器
在springmvc.xml配置文件中,給處理器適配器加入json轉(zhuǎn)換器
<!--處理器適配器 -->
RESTful支持
什么是restful?
Restful就是一個(gè)資源定位及資源操作的風(fēng)格。不是標(biāo)準(zhǔn)也不是協(xié)議,只是一種風(fēng)格。基于這個(gè)風(fēng)格設(shè)計(jì)的軟件可以更簡(jiǎn)潔,更有層次,更易于實(shí)現(xiàn)緩存等機(jī)制
傳統(tǒng)方式操作資源
http://127.0.0.1/item/queryItem.action?id=1查詢,GET
http://127.0.0.1/item/saveItem.action新增,POST
http://127.0.0.1/item/updateItem.action更新,POST
http://127.0.0.1/item/deleteItem.action?id=1刪除,GET或POST
使用RESTful操作資源
http://127.0.0.1/item/1查詢,GET
http://127.0.0.1/item新增,POST
http://127.0.0.1/item更新,PUT
http://127.0.0.1/item/1刪除,DELETE
測(cè)試
首先我們指定參數(shù)注入在地址中,在地址中的參數(shù)使用{}包裹并使用@PathVariable指定地址中的參數(shù),對(duì)應(yīng)到方法的形參中
/**? ? * 通過id查詢商品? ? *? ? *@return*/@RequestMapping("/itemEdit/{id}.html")publicModelAndViewqueryItemByIdRestFul(@PathVariable Integer id){? ? ? ? Items items = itemService.queryItemById(id);//傳遞結(jié)果回頁面ModelAndView mva =newModelAndView();? ? ? ? mva.setViewName("editItem");//設(shè)置數(shù)據(jù)mva.addObject("item", items);returnmva;? ? }
訪問
訪問地址
攔截器
定義
SpringMVC 的處理器攔截器類似于Servlet開發(fā)中的過濾器Filter,用于對(duì)處理器進(jìn)行預(yù)處理和后處理
publicclassHandlerInterceptor1implementsHandlerInterceptor{// controller執(zhí)行后且視圖返回后調(diào)用此方法// 這里可得到執(zhí)行controller時(shí)的異常信息// 這里可記錄操作日志@OverridepublicvoidafterCompletion(HttpServletRequest arg0, HttpServletResponse arg1, Object arg2, Exception arg3)throwsException{? ? ? ? System.out.println("HandlerInterceptor1....afterCompletion");? ? }// controller執(zhí)行后但未返回視圖前調(diào)用此方法// 這里可在返回用戶前對(duì)模型數(shù)據(jù)進(jìn)行加工處理,比如這里加入公用信息以便頁面顯示@OverridepublicvoidpostHandle(HttpServletRequest arg0, HttpServletResponse arg1, Object arg2, ModelAndView arg3)throwsException{? ? ? ? System.out.println("HandlerInterceptor1....postHandle");? ? }// Controller執(zhí)行前調(diào)用此方法// 返回true表示繼續(xù)執(zhí)行,返回false中止執(zhí)行// 這里可以加入登錄校驗(yàn)、權(quán)限攔截等@OverridepublicbooleanpreHandle(HttpServletRequest arg0, HttpServletResponse arg1, Object arg2)throwsException{? ? ? ? System.out.println("HandlerInterceptor1....preHandle");// 設(shè)置為true,測(cè)試使用returntrue;? ? }}
攔截器定義 springmvc.xml
<!-- 配置攔截器 --><!-- 所有的請(qǐng)求都進(jìn)入攔截器 --><!-- 配置具體的攔截器 --><!-- 所有的請(qǐng)求都進(jìn)入攔截器 --><!-- 配置具體的攔截器 -->
多個(gè)攔截器調(diào)用總結(jié)
按照配置執(zhí)行后,我們可以看到 preHandler方法返回false后第一個(gè)攔截器只執(zhí)行了preHandler方法,其他的兩個(gè)方法沒有執(zhí)行,第二個(gè)攔截器的所有方法不執(zhí)行,且Controller也不執(zhí)行了
HandlerInterceptor1的preHandler方法返回true HandlerInterceptor2返回false時(shí),我們可以看到第二個(gè)攔截器的preHandler方法返回false后第一個(gè)攔截器的postHandler沒有執(zhí)行,第二個(gè)攔截器的postHandler和afterCompletion沒有執(zhí)行,且controller也不執(zhí)行了
preHandler按攔截器定義的順序調(diào)用
postHandler按攔截器定義的順序逆序調(diào)用
afterCompletion按攔截器定義順序逆序調(diào)用
postHandler在攔截器鏈內(nèi)所有攔截器返回成功調(diào)用,在多個(gè)攔截器時(shí),只有所有攔截器的preHandler返回成功時(shí) 才會(huì)執(zhí)行postHandler方法
afterCompletion只有preHandle 返回true才調(diào)用 在多個(gè)攔截器時(shí),只要當(dāng)前
作者:So_ProbuING
鏈接:http://www.lxweimin.com/p/b15f5646031b
來源:簡(jiǎn)書
簡(jiǎn)書著作權(quán)歸作者所有,任何形式的轉(zhuǎn)載都請(qǐng)聯(lián)系作者獲得授權(quán)并注明出處。