1. 程式人生 > >學習SpirngMVC之如何獲取請求引數

學習SpirngMVC之如何獲取請求引數

  @RequestParam,你一定見過;@PathVariable,你肯定也知道;@QueryParam,你怎麼會不曉得?!還有你熟悉的他(@CookieValue)!她(@ModelAndView)!它(@ModelAttribute)!沒錯,僅註解這塊,spring mvc就為你打開了五彩斑斕的世界。來來來,不要興(mi)奮(hu),坐下來,我們好好聊聊這麼些個註解兄弟們~~~(wait, 都沒有聽過? 好,來,你坐前排,就你!)

一、spring mvc如何匹配請求路徑——“請求路徑哪家強,RequestMapping名遠揚”

  @RequestMapping是用來對映請求的,比如get請求,post請求,或者REST風格與非REST風格的。 該註解可以用在類上或者方法上,如果用於類上,表示該類中所有方法的父路徑。

  舉例(這裡用到的測試類如SpringMVCTest以及一些頁面在第一篇《學習SpringMVC——從HelloWorld開始》中已經介紹):

  SpringMVCTest.java中加入測試方法:

1 2 3 4 5 @RequestMapping("/testRequestMapping") public String testRequestMapping(){ System.out.println("testRequestMapping"); return SUCCESS; }

  注意這裡在方法級別上添加了註解@RequestMapping(“/testRequestMapping”),  表示可以通過“/testRequestMapping”相對路徑來定位到這個方法,同時我們在SpringMVCTest類上也放了一個類級別的RequestMapping的註解:

1 2 3 @RequestMapping("/springmvc") @Controller public class SpringMVCTest {

  注意這裡還添加了一個@Controller的註解,該註解在SpringMVC 中,負責處理由DispatcherServlet 分發的請求,它把使用者請求的資料經過業務處理層處理之後封裝成一個Model ,然後再把該Model 返回給對應的View 進行展示。至此有了一個“springmvc/testRequestMapping”這樣的路徑,我們就能夠定位到testRequestMapping這個方法上,然後執行方法內的方法體。

  再補充一點,RequestMapping可以實現模糊匹配路徑,比如:

  ?:匹配一個字元

  *:匹配任意字元

  **:匹配多層路徑

  /springmvc/**/lastTest 就可以匹配/springmvc/firstTest/secondTest/lastTest這樣的路徑

二、spring mvc如何獲取請求的引數——“八仙過海,各顯神通”

  1. @PathVariable

  該註解用來對映請求URL中繫結的佔位符。通過@PathVariable可以將URL中佔位符的引數繫結到controller處理方法的入參中,沒聽懂?看例子:

1 2 3 4 5 @RequestMapping("/testPathVariable/{id}") public String testPathVariable(@PathVariable(value="id") Integer id){ System.out.println("testPathVariable:" + id); return SUCCESS; }

  在index.jsp中我們新增一條連線,用來觸發一個請求:

<a href="springmvc/testPathVariable/1">testPathVariable</a><br/><br/>

  我們可以看到這裡有一個超連結,點選後會進入到springmvc/testPathVariable/1對應的controller處理的方法中,那我們現在就是想獲取到這個請求引數中的“1”,所以在testPathVariable方法上加入“/testPathVariable/id,id”,關於{id}的具體對應在該方法的引數中,通過@PathVariable(value="id")來宣告要接收的請求引數,並通過Integer id來繫結和接收。通過該種方式,我們就可以得到前臺頁面請求的引數“1”。

  2. @RequestParam

  該註解也是用來獲取請求引數的。那麼該註解和@PathVariable有何不同呢? 還是看例子:

  在SpringMVCTest中新增方法

1 2 3 4 5 @RequestMapping(value="/testRequestParam") public String testRequestParam(@RequestParam(value="username") String username, @RequestParam(value="age", required=false, defaultValue="0"int age){ System.out.println("testRequestParam" " username:" + username + " age:" +age); return SUCCESS; }

  在index.jsp新增超連結標籤

1 <a href="springmvc/testRequestParam?username=jackie&age=12">testRequestParam</a><br/><br/>

  點選頁面上的超連結,就會匹配controller中testRequestParam方法上的RequestMapping的路徑。注意在該方法中,我們通過@RequestParam這個註解聲明瞭兩個變數,用來獲取請求中query所帶的引數值,一個是username後的值,另一個是age後面的值。

  看到這裡,你大概已經明白了@PathVariable和@RequestParam之間的一些區別了吧,對於像“springmvc/testPathVariable/1”這樣的請求,我們通過@PathVariable來繫結請求的引數;而對於類似“springmvc/testRequestParam?username=jackie&age=12”這樣的請求引數是以鍵值對出現的,我們通過@RequestParam來獲取到如username或age後的具體請求值。

  與RequestParam有異曲同工用法的還有QueryParam,因其不是spring mvc框架內的註解,這裡不再詳述。

  對於不同的請求型別和請求方式,spring mvc都有一套針對的解決方案,下面我們來看看當下比較流行的REST風格的請求是啥樣的——利用REST風格實現增刪改查。

  在SpringMVCTest類中自下而上的實現了查(get)增(post)刪(delete)和改(put)的介面

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 @RequestMapping(value="/testRest/{id}", method=RequestMethod.PUT) public String testRestPut(@PathVariable(value="id") Integer id){ System.out.println("test put:" + id); return SUCCESS; } @RequestMapping(value="/testRest/{id}", method=RequestMethod.DELETE) public String testRestDelete(@PathVariable(value="id") Integer id){ System.out.println("test delete:" + id); return SUCCESS; } @RequestMapping(value="/testRest", method=RequestMethod.POST) public String testRest(){ System.out.println("test post"); return SUCCESS; } @RequestMapping(value="/testRest/{id}", method=RequestMethod.GET) public String testRest(@PathVariable(value="id") Integer id){ System.out.println("test get:" + id); return SUCCESS; }

  那麼前臺介面如何實現呢,相對應的順序為

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 <form action="springmvc/testRest/1" method="post"> <input type="hidden" name="_method" value= "PUT"/> <input type="submit" value="testRestPut"/> </form><br/><br/> <form action="springmvc/testRest/1" method="post"> <input type="hidden" name="_method" value="DELETE"/> <input type="submit" value="TestRest DELETE"/> </form><br><br> <form action="springmvc/testRest" method="post"> <input type="submit" value="testRestPost"> </form><br/><br/> <a href="springmvc/testRest/1">testRest</a><br/><br/>

  除此之外,我們還需要在配置檔案web.xml中新增支援將post轉化為delete和put請求的宣告

1 2 3 4 5 6 7 8 9 10 <!-- 配置HiddenHttpMethodFilter:可以把POST請求轉為DELETE或POST請求 --> <filter> <filter-name>HiddenHttpMethodFilter</filter-name> <filter-class>org.springframework.web.filter.HiddenHttpMethodFilter</filter-class> </filter> <filter-mapping> <filter-name>HiddenHttpMethodFilter</filter-name> <url-pattern>/*</url-pattern> </filter-mapping>

  如你所見,這裡的改和刪都是通過post的方式傳送出去的,因為這裡不支援put和delete來直接實現刪改,而是通過藉助post方式,並悄悄的帶上一塊令牌hidden型別的input標籤來告訴後臺我在前臺傳送的實際上是刪和改的請求。

  那麼這個過程時如何實現的呢,為什麼加上

1 <input type="hidden" name="_method" value="DELETE"/>

這塊令牌,人家後臺就要買你的賬呢。那我們就來看看後來是如何買賬的吧。

  歸根到底還是得益於新增在web.xml中的HiddenHttpMethodFilter這個類,在該類中有一個方法doFilterInternal, 通過除錯我們可以發現其中端倪,啟動tomcat(不能是tomcat8),點選delete操作對應的input標籤,進入除錯介面,我們可以看到:

  • 通過request.getParameter(this.methodParam)在request域中得到this.methodParam(_method)的值,對應於刪除delete的操作,在頁面上,delete中聲明瞭一個hidden的input,其中name就是“_method”,value就是DELETE,所以這裡得到的paramValue的值為“DELETE”
  • 繼續執行,可以看到通過request.getMethod的取值是否與“POST”相等,顯然,這裡是相等,因為我們在前臺頁面中對於delete的操作請求中method宣告為post方式
  • 再往後就是將獲取到的請求方法封裝HttpServletRequest中,完成後續的處理。這裡我們應該明白了為什麼前臺要加上那樣一個hidden的input了。

  小坑:這裡注意啟動不能是tomcat8,而只能是比8小的版本,如7或6等,下圖展示了用tomcat的報錯資訊和用7的成功響應:

總結下,如何傳送put和delete的請求:

  1. 在web.xml中配置HiddenHttpMethodFilter
  2. 傳送post請求
  3. 請求中是個隱藏域,name為”_mothod”,value為put或delete

最後再來說下@CookieValue這個註解。

  3. @CookieValue

  該註解也是差不多的套路,也是一種對映,對映的是一個Cookie值。

  在我們傳送一個請求時,我們可以看到請求中攜帶了一些cookie值

  比如這裡的JSESSIONID或者Path等。現在我們就寫個方法用於獲取Cookie值。

  在SpringMVCTest中新增

1 2 3 4 5 @RequestMapping(value="/testCookieValue") public String testCookieValue(@CookieValue("JSESSIONID") String cookieValue){ System.out.println("testCookieValue: " + cookieValue); return SUCCESS; }

  index.jsp介面上新增連結

1 <a href="springmvc/testCookieValue">testCookieValue</a><br/><br/>

  這樣我們就可以得到類似“testCookieValue: 1410F05C9ADD84E8659C2AC79E8CC666”這樣的結果。

至此,我們介紹了

  1. @RequestMapping的用法
  2. 獲取請求引數的@PathVariable、@RequestParam的用法
  3. 介紹如何實現REST風格的請求,並分析了post如何轉化為delete和put請求
  4. 介紹了@CookieValue的用法