天天看點

SpringMVC 簡單執行個體

SpringMVC是一個基于DispatcherServlet的MVC架構,每一個請求最先通路的都是DispatcherServlet,DispatcherServlet負責轉發每一個Request請求給相應的Handler,Handler處理以後再傳回相應的視圖(View)和模型(Model),傳回的視圖和模型都可以不指定,即可以隻傳回Model或隻傳回View或都不傳回。

DispatcherServlet是繼承自HttpServlet的,既然SpringMVC是基于DispatcherServlet的,那麼我們先來配置一下DispatcherServlet,好讓它能夠管理我們希望它管理的内容。HttpServlet是在web.xml檔案中聲明的。

[html] view plain copy print ?

  1. <servlet>
  2. <servlet-name>securityWeb</servlet-name>
  3. <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
  4. <init-param>
  5. <param-name>contextConfigLocation</param-name>
  6. <param-value>classpath*:conf/applicationContext-mvc.xml</param-value>
  7. </init-param>
  8. <load-on-startup>2</load-on-startup>
  9. </servlet>
  10. <servlet-mapping>
  11. <servlet-name>securityWeb</servlet-name>
  12. <url-pattern>*.do</url-pattern>
  13. </servlet-mapping>
<servlet>
    <servlet-name>securityWeb</servlet-name>
    <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
    <init-param>
      <param-name>contextConfigLocation</param-name>
      <param-value>classpath*:conf/applicationContext-mvc.xml</param-value>
    </init-param>
    <load-on-startup>2</load-on-startup>
  </servlet>
  <servlet-mapping>
    <servlet-name>securityWeb</servlet-name>
    <url-pattern>*.do</url-pattern>
  </servlet-mapping>
           

上面聲明了一個名為securityWeb的DispatcherServlet,該Servlet将處理所有以“.do”結尾的請求。在初始化DispatcherServlet的時候,SpringMVC預設會到/WEB-INF目錄下尋找一個叫[servlet-name]-servlet.xml的配置檔案,來初始化裡面的bean對象,該檔案中對應的bean對象會覆寫spring配置檔案中聲明的同名的bean對象。如上面的就會在/WEB-INF目錄下尋找一個叫securityWeb-servlet.xml的檔案;當然也可以在Servlet中聲明配置檔案的位置。如上面代碼已經寫出路徑。

DispatcherServlet會利用一些特殊的bean來處理Request請求和生成相應的視圖傳回。

關于視圖的傳回,Controller隻負責傳回來一個值,然後到底傳回的是什麼視圖,是由視圖解析器控制的,在jsp中常用的視圖解析器是InternalResourceViewResovler,它會要求一個字首和一個字尾

[html] view plain copy print ?

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <!--
  3. - DispatcherServlet application context for DeclareOnline's web tier.
  4. spring mvc 的 DispatcherServlet 對應的配置檔案
  5. -->
  6. <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  7. xmlns:p="http://www.springframework.org/schema/p" xmlns:context="http://www.springframework.org/schema/context"
  8. xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
  9. http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd" default-autowire="byName">
  10. <!--
  11. - The controllers are autodetected POJOs labeled with the @Controller annotation.
  12. Spring mvc将在該路徑下尋找Controller(有Controller注釋的類)
  13. -->
  14. <context:component-scan base-package="com.zhjy.dps.*"/>
  15. <!-- 使用依賴控制器類名字的慣例,将URI映射到控制器 -->
  16. <bean class="org.springframework.web.servlet.mvc.support.ControllerClassNameHandlerMapping" />
  17. <bean class="org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping" />
  18. <!-- 啟動Spring MVC的注解功能,完成請求和注解POJO的映射 -->
  19. <bean class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter" />
  20. <!-- 定義Spring MVC 的模闆檔案 -->
  21. <bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver">
  22. <!-- 支援JSTL -->
  23. <property name="viewClass" value="org.springframework.web.servlet.view.JstlView"/>
  24. <!-- 模闆的路徑 -->
  25. <property name="prefix" value="/" />
  26. <!-- 模闆檔案字尾 -->
  27. <property name="suffix" value=".jsp" />
  28. </bean>
  29. </beans>
<?xml version="1.0" encoding="UTF-8"?>
<!--
	- DispatcherServlet application context for DeclareOnline's web tier.
	
	spring mvc 的 DispatcherServlet 對應的配置檔案
-->
<beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
		xmlns:p="http://www.springframework.org/schema/p" xmlns:context="http://www.springframework.org/schema/context"
		xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
				http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd"  default-autowire="byName">

	<!--
		- The controllers are autodetected POJOs labeled with the @Controller annotation.
		Spring mvc将在該路徑下尋找Controller(有Controller注釋的類)
	-->
	<context:component-scan base-package="com.zhjy.dps.*"/>
	
	<!-- 使用依賴控制器類名字的慣例,将URI映射到控制器 -->
	<bean class="org.springframework.web.servlet.mvc.support.ControllerClassNameHandlerMapping" />
	
	<bean class="org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping" />
	
	<!-- 啟動Spring MVC的注解功能,完成請求和注解POJO的映射 -->
	<bean class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter" />
	
	<!-- 定義Spring MVC 的模闆檔案 -->
	<bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver">
		<!-- 支援JSTL -->
		<property name="viewClass" value="org.springframework.web.servlet.view.JstlView"/>
		<!-- 模闆的路徑 -->
		<property name="prefix" value="/" />
		<!-- 模闆檔案字尾 -->
		<property name="suffix" value=".jsp" />
	</bean>
	
	
</beans>
           

在上述視圖解析器中,如果Controller傳回的是securityWeb/index,那麼通過視圖解析器解析之後的視圖就是/WEB-INF/securityWeb/index.jsp。

要使用注解的SpringMVC需要在SpringMVC的配置檔案中進行聲明,具體方式為先引入mvc命名空間,然後利用<mvc:annotation-driven />進行聲明。

主要是說說Controller.

一個類使用了@Controller進行标記的都是Controller

[java] view plain copy print ?

  1. @Controller
  2. public class PlanController {
  3. }
@Controller
public class PlanController {
}
           

有了Controller之後,那麼到底是怎樣請求一個Controller具體的方法的呢,那是通過@RequestMapping來标記的,@RequestMapping可以标記在類上面,也可以标記在方法上,當方法上和類上都标記了@RequestMapping的時候,那麼對應的方法對應的Url就是類上的加方法上的,如下面的index方法,其對應的URL應為類上的/blog加上index方法上的/index,是以應為/blog/index,是以當請求/blog/index.do的時候就會通路BlogController的index方法。

[java] view plain copy print ?

  1. @Controller
  2. public class PlanController {
  3. @RequestMapping("service/index")
  4. public String index(Map<String, Object> map) {
  5. return "secruityWeb/index";
  6. }
  7. }
@Controller  
public class PlanController {  
  
      
    @RequestMapping("service/index")  
    public String index(Map<String, Object> map) {  
        return "secruityWeb/index";  
    }  
}  
           

在RequestMapping中還可以指定一個屬性method,其主要對應的值有RequestMethod.GET和RequestMethod.POST,利用該屬性可以嚴格的控制某一方法隻能被标記的請求路徑對應的請求方法才能通路,如指定method的值為GET,則表示隻有通過GET方式才能通路該方法,預設是都可以通路。

在SpringMVC中常用的注解還有@PathVariable,@RequestParam,@PathVariable标記在方法的參數上,利用它标記的參數可以利用請求路徑傳值,看下面一個例子

[java] view plain copy print ?

  1. @RequestMapping(value="/comment/{blogId}", method=RequestMethod.POST)
  2. public void comment(Comment comment,@PathVariable int blogId, HttpSession session, HttpServletResponse response) throws IOException {
  3. }
@RequestMapping(value="/comment/{blogId}", method=RequestMethod.POST)  
public void comment(Comment comment,@PathVariable int blogId, HttpSession session, HttpServletResponse response) throws IOException {  
      
}  
           

在該例子中,blogId是被@PathVariable标記為請求路徑變量的,如果請求的是/blog/comment/1.do的時候就表示blogId的值為1. 同樣@RequestParam也是用來給參數傳值的,但是它是從頭request的參數裡面取值,相當于request.getParameter("參數名")方法。

在Controller的方法中,如果需要WEB元素HttpServletRequest,HttpServletResponse和HttpSession,隻需要在給方法一個對應的參數,那麼在通路的時候SpringMVC就會自動給其傳值,但是需要注意的是在傳入Session的時候如果是第一次通路系統的時候就調用session會報錯,因為這個時候session還沒有生成。

接下來讨論一下方法的傳回值,主要有一下情況:

  • 傳回一個ModelAndView,其中Model是一個Map,裡面存放的是一對對的鍵值對,其可以直接在頁面上使用,View是一個字元串,表示的是某一個View的名稱
  • 傳回一個View,也就是一個字元串,這個時候如果需要給頁面傳值,可以給方法一個Map參數,該Map就相當于一個Model,往該Model裡面存入鍵值對就可以在頁面上進行通路了
  • 傳回一個Model也就是一個Map,這個時候将解析預設的生成的view name。
  • 什麼也不傳回,這個時候可以利用HttpServletResponse進行傳回,也可以直接使用printStream進行傳回

下面是一個執行個體:

[java] view plain copy print ?

  1. @RequestMapping("/work/plan/findProjectByState")
  2. public String findProjectByState(YearPlanMgrQueryForm form,String planState, HttpServletRequest request, HttpServletResponse response) {
  3. HttpSession session = request.getSession();
  4. String industryId = (String) session.getAttribute("industryID");
  5. String planName = request.getParameter("planName");
  6. String personSituation = request.getParameter("planState");
  7. try {
  8. if(planState != null && !"".equals(planState)){
  9. form.setPlanState(planState);
  10. }
  11. int pageNo = 1;
  12. int fetch = 8;
  13. String pageNumber = request.getParameter("pageNumber");
  14. String pageSize = request.getParameter("pageSize");
  15. // 得到頁碼
  16. if (pageNumber == null) {
  17. pageNo = PageUtil.getPageNum(request, PageUtil.DEFAULT_PAGENUM);
  18. } else {
  19. pageNo = Integer.parseInt(pageNumber);
  20. }
  21. // 得到每頁記錄數
  22. if (pageSize == null) {
  23. fetch = PageUtil.getPageSize(request, 8);
  24. } else {
  25. fetch = Integer.parseInt(pageSize);
  26. }
  27. // 擷取清單
  28. System.out.println(personSituation);
  29. Page<Map<String, Object>> page = projectPlanService.findProjectByState(planName, industryId, personSituation, pageNo, fetch);
  30. request.setAttribute("personSituation", personSituation);
  31. request.setAttribute("page", page);
  32. request.setAttribute("pageSize", fetch);
  33. } catch (Exception e) {
  34. throw new PlatformException(e);
  35. }
  36. return "/module/industry/planlist";
  37. }