基於註解的SpringMVC簡單介紹
SpringMVC是一個基於DispatcherServlet的MVC框架,每一個請求最先訪問的都是DispatcherServlet,DispatcherServlet負責轉發每一個Request請求給相應的Handler,Handler處理以後再返回相應的檢視(View)和模型(Model),返回的檢視和模型都可以不指定,即可以只返回Model或只返回View或都不返回。在使用註解的SpringMVC中,處理器Handler是基於@Controller和@RequestMapping這兩個註解的,@Controller宣告一個處理器類,@RequestMapping宣告對應請求的對映關係,這樣就可以提供一個非常靈活的匹配和處理方式。
DispatcherServlet是繼承自HttpServlet的,既然SpringMVC是基於DispatcherServlet的,那麼我們先來配置一下DispatcherServlet,好讓它能夠管理我們希望它管理的內容。HttpServlet是在web.xml檔案中宣告的。
Xml程式碼- <servlet>
- <servlet-name>blog</servlet-name>
-
<servlet-class>org.springframework.web.servlet.DispatcherServlet</
- <load-on-startup>1</load-on-startup>
- </servlet>
- <servlet-mapping>
- <servlet-name>blog</servlet-name>
- <url-pattern>*.do</url-pattern>
- </servlet-mapping>
上面聲明瞭一個名為blog的DispatcherServlet,該Servlet將處理所有以“.do”結尾的請求。在初始化DispatcherServlet的時候,SpringMVC預設會到/WEB-INF目錄下尋找一個叫[servlet-name]-servlet.xml的配置檔案,來初始化裡面的bean物件,該檔案中對應的bean物件會覆蓋spring配置檔案中宣告的同名的bean物件。如上面的就會在/WEB-INF目錄下尋找一個叫blog-servlet.xml的檔案;當然也可以在Servlet中宣告配置檔案的位置,那就是通過Servlet的初始化引數來設定contextConfigLocation引數的值。
- <servlet>
- <servlet-name>blog</servlet-name>
- <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
- <init-param>
- <param-name>contextConfigLocation</param-name>
- <param-value>/WEB-INF/blog-servlet.xml</param-value>
- </init-param>
- <load-on-startup>1</load-on-startup>
- </servlet>
- <servlet-mapping>
- <servlet-name>blog</servlet-name>
- <url-pattern>*.do</url-pattern>
- </servlet-mapping>
DispatcherServlet會利用一些特殊的bean來處理Request請求和生成相應的檢視返回。
關於檢視的返回,Controller只負責傳回來一個值,然後到底返回的是什麼檢視,是由檢視解析器控制的,在jsp中常用的檢視解析器是InternalResourceViewResovler,它會要求一個字首和一個字尾
Xml程式碼- <bean
- class="org.springframework.web.servlet.view.InternalResourceViewResolver">
- <propertyname="prefix"value="/WEB-INF/"/>
- <propertyname="suffix"value=".jsp"/>
- </bean>
在上述檢視解析器中,如果Controller返回的是blog/index,那麼通過檢視解析器解析之後的檢視就是/WEB-INF/blog/index.jsp。
要使用註解的SpringMVC需要在SpringMVC的配置檔案中進行宣告,具體方式為先引入mvc名稱空間,然後利用<mvc:annotation-driven />進行宣告。
Xml程式碼- <beansxmlns="http://www.springframework.org/schema/beans"
- xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xmlns:context="http://www.springframework.org/schema/context"
- <spanstyle="background-color: #00ff00;"><spanstyle="color: #ff0000;">xmlns:mvc="http://www.springframework.org/schema/mvc"</span>
- </span>
- xsi:schemaLocation="http://www.springframework.org/schema/beans
- http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
- http://www.springframework.org/schema/context
- http://www.springframework.org/schema/context/spring-context-3.0.xsd
- <spanstyle="background-color: #00ff00; color: #ff0000;"> http://www.springframework.org/schema/mvc
- http://www.springframework.org/schema/mvc/spring-mvc-3.0.xsd"</span>
- >
- <mvc:annotation-driven/>
- </beans>
主要是說說Controller.
在SpringMVC中Controller不需要繼承什麼類,也不需要實現什麼介面,一切使用了@Controller進行標記的類都是Controller
Java程式碼- @Controller
- publicclass BlogController {
- }
有了Controller之後,那麼到底是怎樣請求一個Controller具體的方法的呢,那是通過@RequestMapping來標記的,@RequestMapping可以標記在類上面,也可以標記在方法上,當方法上和類上都標記了@RequestMapping的時候,那麼對應的方法對應的Url就是類上的加方法上的,如下面的index方法,其對應的URL應為類上的/blog加上index方法上的/index,所以應為/blog/index,所以當請求/blog/index.do的時候就會訪問BlogController的index方法。加在類上的@RequestMapping不是必須的,當Controller類上加上了@RequestMapping的時候,那麼Controller方法上的@RequestMapping就是相對於類上的@RequestMapping而言的,也就是前面說的請求對映的時候是類上的地址加方法上的地址,而當Controller類上沒有加@RequestMapping的時候,方法上的@RequestMapping就是絕對路徑了。
Java程式碼- @Controller
- @RequestMapping("/blog")
- publicclass BlogController {
- @RequestMapping("/index")
- public String index(Map<String, Object> map) {
- return"blog/index";
- }
- }
在上面的程式碼中,如果index方法上沒有RequestMapping註解,而只有BlogController類上有,且該類只有一個方法的時候,直接請求類上的URL就會呼叫裡面的方法,即直接請求/blog.do的時候就會呼叫index方法。
在RequestMapping中還可以指定一個屬性method,其主要對應的值有RequestMethod.GET和RequestMethod.POST,利用該屬性可以嚴格的控制某一方法只能被標記的請求路徑對應的請求方法才能訪問,如指定method的值為GET,則表示只有通過GET方式才能訪問該方法,預設是都可以訪問。RequestMapping中的URL對映還支援萬用字元*,如:
Java程式碼- @Controller
- @RequestMapping("/blog")
- publicclass BlogController {
- @RequestMapping("/*/index")
- public String index(Map<String, Object> map) {
- return"blog/index";
- }
- }
在@RequestMapping中還有一個屬性params,可以通過該屬性指定請求引數中必須包含某一引數,或必須不包含某一引數,或某引數的值必須是什麼,以此來縮小指定的對映範圍。
Java程式碼- @Controller
- @RequestMapping("/blog")
- publicclass BlogController {
- @RequestMapping(value="/index", params="param1=value1")
- public String index(Map<String, Object> map) {
- return"blog/index";
- }
- }
在上面示例中,只有當請求/blog/index.do並且請求引數param1的值為value1的時候才能訪問到對應的index方法。如果params的值為"param1",則表示請求引數只要包含param1就可以了,至於它的值是什麼無所謂;如果params的值為"!param1",則表示請求引數必須不包含param1才可以。@RequestMapping中還可以使用header來縮小對映範圍,如:
Java程式碼- @Controller
- @RequestMapping("/blog")
- publicclass BlogController {
- @RequestMapping(value="/index",headers="content-type=text/html")
- public String index(Map<String, Object> map) {
- return"blog/index";
- }
- }
在SpringMVC中常用的註解還有@PathVariable,@RequestParam,@PathVariable標記在方法的引數上,利用它標記的引數可以利用請求路徑傳值,看下面一個例子
Java程式碼- @RequestMapping(value="/comment/{blogId}", method=RequestMethod.POST)
- publicvoid comment(Comment comment,@PathVariableint blogId, HttpSession session, HttpServletResponse response) throws IOException {
- }
在該例子中,blogId是被@PathVariable標記為請求路徑變數的,如果請求的是/blog/comment/1.do的時候就表示blogId的值為1,@PathVariable在進行賦值的時候如果像上面那樣沒有指定後面接的變數是對應URL中的哪個變數時預設是從URL中取跟後面接的變數名相同的變數,如上面示例中的@PathVariable int blogId,沒有指明要獲取URL中的哪個變數,這個時候就預設取URL中的blogId變數對應的值賦給方法引數中的blogId,那如果方法引數的名稱跟RequestMapping中定義的訪問路徑中的變數名不一樣,或者我要利用PathVariable明確指定後面接的方法引數是對應於URL中的哪個變數時,可以像下面這樣做,在PathVariable中給定一個value="blogId"(只有一個引數的時候value是可以省略的)值明確指定方法引數中的id變數是對應請求路徑定義中的blogId變數的。
Java程式碼- @RequestMapping(value="/comment/{blogId}", method=RequestMethod.POST)
- publicvoid comment(Comment comment,@PathVariable("blogId") int id, HttpSession session, HttpServletResponse response) throws IOException {
- }
同樣@RequestParam也是用來給引數傳值的,但是它是從頭request的引數裡面取值,相當於request.getParameter("引數名")方法。它的取值規則跟@PathVariable是一樣的,當沒有指定的時候,預設是從request中取名稱跟後面接的變數名同名的引數值,當要明確從request中取一個引數的時候使用@RequestParam("引數名"),如下所示:
Java程式碼- @RequestMapping("/show")
- publicvoid showParam(@RequestParamint id, @RequestParam("name") String username) {
- //這樣做進行URL請求訪問這個方法的時候,就會先從request中獲取引數id的值賦給引數變數id,從request中獲取引數name的值賦給引數變數username
- }
在Controller的方法中,如果需要WEB元素HttpServletRequest,HttpServletResponse和HttpSession,只需要在給方法一個對應的引數,那麼在訪問的時候SpringMVC就會自動給其傳值,但是需要注意的是在傳入Session的時候如果是第一次訪問系統的時候就呼叫session會報錯,因為這個時候session還沒有生成。
接下來討論一下方法的返回值,主要有以下情況:
- 返回一個ModelAndView,其中Model是一個Map,裡面存放的是一對對的鍵值對,其可以直接在頁面上使用,View是一個字串,表示的是某一個View的名稱
- 返回一個字串,這個時候如果需要給頁面傳值,可以給方法一個Map引數,該Map就相當於一個Model,往該Model裡面存入鍵值對就可以在頁面上進行訪問了
- 返回一個View物件
- 返回一個Model也就是一個Map,這個時候將解析預設生成的view name,預設情況view name就是方法名,這裡之前搞錯了,感謝網友的指正。預設的View Name是由RequestToViewNameTranslator來解析的,顧名思義就是把request翻譯成viewName,在沒有指定RequestToViewNameTranslator時,Spring將使用其自身的預設實現DefaultRequestToViewNameTranslator的預設配置,即取到當前請求的URI,去掉最前和最後的斜槓“/”,以及對應的字尾。所以當你請求“http://localhost/app/abc”的時候,對應的預設viewName就是請求URI——“/abc”去掉最前最後的斜槓和字尾之後的結果,即“abc”,請求“http://localhost/app/abc/efg”時對應的預設檢視名稱是“abc/efg”,“http://localhost/app/abc/efg/hi.html”——>“abc/efg/hi”。如果需要改變預設的檢視名稱的解析方式,可以在SpringMVC的配置檔案中配置一個名稱為viewNameTranslator,型別為RequestToViewNameTranslator的bean。如果該bean是DefaultRequestToViewNameTranslator,那麼你可以通過prefix屬性指定檢視名稱的字首,通過suffix指定字尾,通過stripLeadingSlash指定是否需要去掉最前面的斜槓,更多可指定的屬性請參考DefaultRequestToViewNameTranslator的實現。當然你也可以定義自己的RequestToViewNameTranslator實現類,實現RequestToViewNameTranslator介面的getViewName(HttpServletRequest request)方法,實現自己的獲取預設檢視名稱的邏輯。
- 什麼也不返回,這個時候可以在方法體中直接往HttpServletResponse寫入返回內容,否則將會由RequestToViewNameTranslator來決定
- 任何其他型別的物件。這個時候就會把該方法返回型別物件當做返回Model模型的一個屬性返回給檢視使用,這個屬性名稱可以通過在方法上給定@ModelAttribute註解來指定,否則將預設使用該返回類名稱作為屬性名稱。
- @RequestMapping("/{owner}/index")
- public String userIndex(Map<String, Object> map,@PathVariable String owner, HttpServletRequest request) throws ParserException {
- List<DefCategory> categories = categoryService.find(owner);
- int offset = Util.getOffset(request);
- Pager<Blog> pager = blogService.find(owner, 0, offset, maxResults);
- int totalRecords = pager.getTotalRecords();
- List<Blog> blogs = pager.getData();
- Util.shortBlog(blogs);
- List<Message> messages = messageService.find(owner, 0, 5).getData();
- Util.shortMessage(messages, 20);
- map.put("messages", messages);
- map.put("totalRecords", totalRecords);
- List<BlogStore> stores = storeService.find(owner, 0, 5).getData();
- map.put("maxResults", maxResults);
- map.put("blogs", blogs);
- map.put("totalRecords", totalRecords);
- map.put("owner", userService.find(owner));
- map.put("defCategories", categories);
- map.put("stores", stores);
- return"blog/userIndex";
- }
給頁面傳值
在Controller中把請求轉發給業務邏輯層進行處理之後需要把業務邏輯層的處理結果進行展現,在展現的過程中就需要我們把處理結果傳給展示層進行展示。那麼處理結果是怎麼進行傳遞的呢?前面已經說了Controller的返回結果可以是一個包含模型和檢視的ModelAndView,也可以僅僅是一個檢視View,當然也可以什麼都不返回,還可以是僅僅返回一個Model。我們知道模型Model是用來封裝資料給檢視View進行展示的,那麼,在SpringMVC中,如果要把我們後臺的資訊傳遞給前臺進行展示的話應該怎麼做呢?這主要有兩種方式:
1.返回包含模型Model的ModelAndView,或者是直接返回一個Model(這個時候就需要考慮預設的檢視),這種型別的返回結果是包含Model的,這個Model物件裡面的對應屬性列都可以直接在視圖裡面使用。
2.如果是直接返回一個檢視View,這個時候SpringMVC提供了一種類似於在Controller方法中獲取HttpRequest物件的機制,這個時候我們只需要給定Controller方法一個Map引數,然後在方法體裡面給這個Map加上需要傳遞到檢視的鍵值對,這樣在檢視中就可以直接訪問對應的鍵值對