springMVC請求映射全面分析
大?。?/span>0.6 MB 人氣: 2017-10-11 需要積分:1
標(biāo)簽:SpringMVC(5691)
在springMVC的控制器中,我們常使用@RequestMapping來完成我們的請求映射,我們可以在類定義上和方法定義上使用注解,其配置的路徑將為類中定義的所有方法的父路徑,如上篇實(shí)例中的/user(類)/hello(方法)。一般的,我們類定義上的路徑注解起到命名空間的作用,防止不同方法的路徑映射產(chǎn)生沖突,比如我在UserController和ArticleController下都定義了如下的方法:
@RequestMapping(“l(fā)ist”) publicvoidlist(){ 。..。 }
一個list映射路徑,這時候springMVC就不知道該將請求交給到哪個方法處理。當(dāng)然,我們也能在方法上進(jìn)行二級路徑配置區(qū)分:
/*************UserController***********/@RequestMapping(“user/list”) publicvoidlist(){ 。..。 } /*************ArticleController***********/@RequestMapping(“article/list”) publicvoidlist(){ 。..。 }
這樣就能有效防止沖突了,但如果我有很多個方法存在這樣的沖突,是否都要在每個方法加上前綴呢?這時候我們可以選擇在類路徑上注解@RequestMapping來對全體方法進(jìn)行區(qū)分。
通過url進(jìn)行映射
1. Ant風(fēng)格字符匹配
除了標(biāo)準(zhǔn)的url外,@RequestMapping還支持Ant風(fēng)格字符,即”?”、”*”、”**”,其中
1. “?”:匹配一個任意字符,如/user/a?,匹配user/aa,user/ab等路徑
2. “*”:匹配任意字符串,如/user/a*,匹配/user下任意以a開頭的路徑如/user/abc,/user/aqw等
3. “**“:匹配多級路徑字符串,如/user/**/list,匹配/user/user1/list,/user/1resu/list等
在這里,需要注意的是當(dāng)*的 位置與 個數(shù)不同時,*可以代表的 字符數(shù)有區(qū)別,看下面示例:
@RequestMapping(“u1/*”)//只能匹配u1/a,u1/b,不能匹配u1/————即此時*表示一個或多個字符publicvoidtest(HttpServletResponse response) throwsIOException{ response.getWriter().print(“u1/*”); } @RequestMapping(“u1/**”)//能夠匹配u1/,u1/qq,u1/qq/ww,這里要特別注意的是,“**“能匹配零個而“*”不能publicvoidtest(HttpServletResponse response) throwsIOException{ response.getWriter().print(“u1/*”); } @RequestMapping(“u2/a*”)//能夠匹配u2/a,u2/ab,u2/aqqqq等————即此時*表示零個或零個以上字符publicvoidtest1(HttpServletResponse response) throwsIOException{ response.getWriter().print(“u2/a*”); }
2. restful占位符匹配
除了使用上面風(fēng)格,@RequestMapping還支持restful風(fēng)格占位符的形式,假如我們需要針對特定用戶查看其特定文章,restful風(fēng)格路徑匹配如下所示:
@Controller//注解為控制器,通過spring容器掃描,會注冊為一個Bean@RequestMapping(“/user/{uid}”)//一級訪問路徑,對類中所有方法生效publicclassUserController{@RequestMapping(“article/{aid}”) publicString detail(@PathVariable(“uid”)Integer uid,@PathVariable(“aid”)Integer aid){ System.out.println( “查看id為”+ uid + “的用戶文章,且文章id為”+aid); return“someplace”; } }
這里,如果我們想訪問用戶id為1,文章id為2的用戶文章,就可以訪問如下路徑:[項(xiàng)目根路徑]/user/1/article/2來完成。
我們使用@PathVariable(“val”)來完成對應(yīng)路徑中{val}的資源請求,這里的兩個val名稱需一致,緊接著的方法入?yún)⒚秩我?,我們剛剛示例了一個多路徑參數(shù)綁定,假設(shè)只有一個,如下也是合法的:
@RequestMapping(“user/{uid}”) publicString detail(@PathVariable(“uid”)Integer notUid){//notUid名字也能成功綁定return“someplace”; }
此外,如果我們?nèi)雲(yún)⒚趾蛈rl路徑資源名稱一致,則可以省略配置@PathVariable中的value值,如下實(shí)例也能正確綁定路徑資源到入?yún)?br /> @RequestMapping(“user/{uid}”) publicString detail(@PathVariable Integer uid){//notUid名字也能成功綁定return“someplace”; }
3. 優(yōu)先匹配規(guī)則
url還有如下兩個常見匹配準(zhǔn)則: 最長最精確優(yōu)先匹配和 占位符優(yōu)先匹配
1. 最長最精確優(yōu)先匹配
下面我們來看前一個匹配實(shí)例:
@RequestMapping(“test/**”) publicvoidtest2(HttpServletResponse response) throwsIOException{ response.getWriter().print(“test/**”); } @RequestMapping(“test/*”) publicvoidtest3(HttpServletResponse response) throwsIOException{ response.getWriter().print(“test/*”); } @RequestMapping(“test/*/**”) publicvoidtest4(HttpServletResponse response) throwsIOException{ response.getWriter().print(“test/*/**”); } @RequestMapping(“test/*/*”) publicvoidtest5(HttpServletResponse response) throwsIOException{ response.getWriter().print(“test/*/*”); } @RequestMapping(“test/1/*”) publicvoidtest6(HttpServletResponse response) throwsIOException{ response.getWriter().print(“test/1/*”); } @RequestMapping(“test/1/2”) publicvoidtest7(HttpServletResponse response) throwsIOException{ response.getWriter().print(“test/1/2”); }
直接看上面匹配會覺得很亂,我們直接看下面的測試:
測試
匹配結(jié)果
test/a 匹配test/*而不匹配test/**(更精確優(yōu)先匹配)
test/a/a/aa/a 匹配test/**而不匹配test/*/**,(在多層匹配中,**比***更精確)
test/a/a 匹配test/*/*,因?yàn)?*/*比**精確
test/1/a 匹配test/1/*,因?yàn)?1/*比/*/*精確
test/1/2 匹配test/1/2,這是完全匹配
2. 占位符優(yōu)先匹配原則
占位符是指@PathVariable等路徑資源占位符,下面我們在看一個實(shí)例
@RequestMapping(“test/1/2”) publicvoidtest7(HttpServletResponse response) throwsIOException{ response.getWriter().print(“test/1/2”); } @RequestMapping(“test/1/{id}”) publicvoidtest8(HttpServletResponse response,@PathVariable Integer id ) throwsIOException{ response.getWriter().print(“test/1/(myId=)”+ id ); } @RequestMapping(“test/1/a”) publicvoidtest7(HttpServletResponse response) throwsIOException{ response.getWriter().print(“test/1/a”); }
從上一個實(shí)例的所有路徑映射中,我們測試出test/1/2是最精確的。但我們根據(jù)添加了占位符映射,在游覽器輸入 test/1/2,此時游覽器返回 test/1/(myId=)2,即 占位符的優(yōu)先級比普通字符串的優(yōu)先級更高!但如果我們此時輸入 test/1/a。程序不會因?yàn)槲覀兊?在方法入?yún)⒅衖d映射為Integer類型而放棄匹配,占位符的優(yōu)先級依然比字符(串)a的優(yōu)先級高,但由于 “a”不能轉(zhuǎn)化為Integer類型,所以服務(wù)器會返回 400錯誤
通過HTTP其它請求資源映射
除了使用url外,我們還能通過請求參數(shù)、請求方法、或請求頭進(jìn)行映射
我們先看看@RequestMapping的完整屬性列表:
屬性
說明
value 指定請求的實(shí)際地址, 比如 /action/info之類。
method 指定請求的method類型, GET、POST、PUT、DELETE等
consumes 指定處理請求的提交內(nèi)容類型(Content-Type),例如application/json, text/html;
produces 指定返回的內(nèi)容類型,僅當(dāng)request請求頭中的(Accept)類型中包含該指定類型才返回
params 指定request中必須包含某些參數(shù)值是,才讓該方法處理
headers 指定request中必須包含某些指定的header值,才能讓該方法處理請求
其中,consumes, produces使用content-type信息進(jìn)行過濾信息;headers中可以使用content-type進(jìn)行過濾和判斷。
在前面的使用中,我們發(fā)現(xiàn)并沒有指定value屬性,直接在括號里輸入字符串也能向value屬性賦值,這是因?yàn)樵?a href='http://ttokpm.com/v/tag/852/' target='_blank' class='arckwlink_none'>java注解中不加其他屬性,直接賦值必定是針對注解的value成員,如果該注解沒有名為value的成員,則會報錯
下面我們先看幾個示例:
示例1:vmethod,headers
@RequestMapping(value = “testa”,method = RequestMethod.POST,headers = “content-type=text/*”)
表示映射路徑為testa,請求方法必須為POST方法(如果我們用post發(fā)出請求,會返回錯誤信息HTTP Status 405 - Request method ‘GET’ not supported),headers部分表示請求頭信息中必須包含等號后相應(yīng)部分內(nèi)容,*匹配任意字符串
示例2:consumes
@RequestMapping(value = “testb”, consumes=“application/json”)
表示方法僅匹配request Content-Type為“application/json”類型的請求。
示例3:produces
@RequestMapping(value = “/testc”, produces=“application/json”)
表示方法匹配的請求需要請求頭中Accept部分包含”application/json“,同時在響應(yīng)時,會將返回內(nèi)容同時設(shè)置為”application/json‘’
示例4:params
@RequestMapping(value = “testd”,method = RequestMethod.GET,params = {“id1”,“id2”}) publicvoidtest12(HttpServletResponse response,Integer id1,Integer id2) throwsIOException{ response.getWriter().print(id1 + “——”+ id2); }
示例表示入?yún)⑿璋瑓?shù)名為id1,id2的兩個參數(shù),這里如果我輸入:
1. http://localhost:8080/springMVC/user/testd—-
2. http://localhost:8080/springMVC/user/testd?id1=1—報404錯誤
3. ttp://localhost:8080/springMVC/user/testd?id1=1&id2=2—-返回1——2
4. ttp://localhost:8080/springMVC/user/testd?id1=1&id2=2&id3=3—-返回1——2
從以上我們可以看出,只有具有相應(yīng)參數(shù)的才能完成映射,且可以有除了params中要求以外的參數(shù),如id3。
在params的常見映射規(guī)則如下:
示例規(guī)則
說明
”param1” 請求必須包含名為param1的參數(shù)
“!param1” 請求中不能包含名為param1的參數(shù)
“param1!=value1 請求中必須包含param1參數(shù),但其值不能為value1
{“param1=value1”,”param2”} 請求中需要包含param1參數(shù)和param2參數(shù),且param1的值必須為value1
?
非常好我支持^.^
(0) 0%
不好我反對
(0) 0%
下載地址
springMVC請求映射全面分析下載
相關(guān)電子資料下載
- Android架構(gòu)模式飛速演進(jìn) 到底哪一個才是自己最需要的? 251
- Spring Boot如何優(yōu)雅實(shí)現(xiàn)數(shù)據(jù)加密存儲、模糊匹配和脫敏 605
- SpringMVC 如何優(yōu)雅的處理各種異常? 147
- Spring Boot中整合兩種定時任務(wù)的方法 879
- SpringBoot與SpringMVC的區(qū)別 315
- spring和springmvc雖是父子,但并不和諧 1316
- idea開發(fā)springmvc非maven版 1054
- springMVC后臺接受前端上傳的文件及下載文件 3168
- springmvc 自定義攔截器實(shí)現(xiàn)未登錄用戶的攔截 2343