作為一個(gè)J2EE的開發(fā)者,我們基本上是開發(fā)基于WEB的應(yīng)用系統(tǒng)。事實(shí)上,工作流,狀態(tài)管理以及驗(yàn)證都是需要解決的重要功能。而HTTP協(xié)議的無狀態(tài)性決定了這些功能都不容易實(shí)現(xiàn)。Spring的WEB框架就是來幫助我們解決這些問題的。使用Spring,我們可以讓WEB框架自動(dòng)將傳遞進(jìn)來的請求參數(shù)填充到模型對象中,同時(shí)提供驗(yàn)證和錯(cuò)誤處理。還可以管理用戶在WEB表單中創(chuàng)建的對象的狀態(tài)。在了解SpringMVC之前,我們先來復(fù)習(xí)下MVC.
MVC從邏輯上把應(yīng)用分為模型組件,視圖組件和控制器組件。其中控制器組件又可以細(xì)分為:前端控制器組件和后端控制器組件。
我們來看一下MVC的基本工作流程:
首先是客戶端(通常是瀏覽器)發(fā)出一個(gè)請求。第一個(gè)接受這個(gè)請求的組件一般是一個(gè)前端控制器。它將不同的請求交給不同的后端控制器來處理,而在后端控制器里面又可以調(diào)用相應(yīng)的模型對象來處理具體的業(yè)務(wù)邏輯,最后再返回一個(gè)特定的視圖響應(yīng)給客戶端。
怎么理解MVC呢? 我們舉一個(gè)現(xiàn)實(shí)的例子,前段時(shí)間地震太可怕了,全國人民都在上下一心抗震救災(zāi)。我們敬愛的溫總理給武警司令下命令讓他完成抗震救災(zāi)的艱巨任務(wù),于是武警司 令就根據(jù)各地情況派遣不同類型的特種兵到不同的地區(qū)去,然后特種兵使用大大小小的工具完成了任務(wù),最后上交給司令一份統(tǒng)計(jì)圖表,司令再上交給溫總理。我們 就來分析一下,這符不符合MVC的設(shè)計(jì)模式。這里呢,溫總理就是客戶端,武警司令就是前端控制器,特種兵就是后端控制器,特種兵所使用的工具就是模型,最后上交的統(tǒng)計(jì)圖表就是視圖。
現(xiàn)在大家應(yīng)該可以理解MVC的設(shè)計(jì)思想了。拿我們比較熟悉的struts框架來說,前端控制器就是ActionServlet,后端控制器就是Action.請求的URL和后端控制器的映射關(guān)系在struts-config.xml上的<action-mappings>里面配置,模型對象就是我們平常寫的DAO/DTO,返回的視圖類型一般就是jsp了。
那么,在spring MVC中又是怎樣的呢?,我們先按上面對號(hào)入座地說一下。第一個(gè)接受這個(gè)請求的前端控制器叫DispatcherServlet,后端控制器叫Controller。負(fù)責(zé)處理請求URL和后端控制器映射的叫HandMapping,它有多種類型,比較靈活,也是在一個(gè)xml文件上進(jìn)行配置。負(fù)責(zé)業(yè)務(wù)邏輯處理的模型對象一般也是我們平常寫的DAO/DTO組件。只是它最后的返回更靈活,Controller返回一個(gè)ModelAndView對象給DispatcherServlet,ModelAndView可以攜帶一個(gè)視圖對象,也可以攜帶一個(gè)視圖對象的邏輯名。如果攜帶的是一個(gè)視圖對象的邏輯名,那DispatcherServlet需要一個(gè)ViewResolver來查找用于渲染回應(yīng)的視圖對象。最后,DispatcherServlet將請求分派給ModelAndView對象指定的視圖對象。視圖對象負(fù)責(zé)渲染返回給客戶的回應(yīng)。
我們學(xué)習(xí)知識(shí),一樣要注意運(yùn)用類比遷移的方法。有人說,我學(xué)會(huì)了struts,但學(xué)不會(huì)springMVC。這是說不過去的。因?yàn)槎叩乃枷胪耆恢隆K裕覀冎恍枰獙W(xué)習(xí)它們不一樣的地方就可以啦。好,我們現(xiàn)在先跑一個(gè)最簡單的例子,讓大家對構(gòu)建SpringMVC的基本步驟有個(gè)直觀的認(rèn)識(shí)。大家要邊跑邊跟struts做比較。
(1)建立動(dòng)態(tài)web工程,導(dǎo)入spring的jar包。
(2)配置DispatcherServlet
DispatcherServlet是SpringMVC的核心,將下面Servlet的注冊信息登記在web.xml中。一定要記住:Servlet跟伴侶一樣,要結(jié)婚,先得注冊登記!
<servlet>
<servlet-name>test</servlet-name> <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>test</servlet-name>
<url-pattern>*.do</url-pattern>
</servlet-mapping>
(3)編寫Controller,做核心配置文件,并配置url和Controller的映射
package com.wepull.test;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.springframework.web.servlet.ModelAndView;
import org.springframework.web.servlet.mvc.Controller;
public classHelloController implements Controller{
public ModelAndView handleRequest(HttpServletRequest request,
HttpServletResponse response) throws Exception {
request.setAttribute("hello", "welcome to spring!");
return new ModelAndView("welcome");
}
}
我們知道Struts有個(gè)XML格式的核心配置文件,springMVC當(dāng)然也有,在WEB-INF下新建一XML文件:test-servlet.xml.注意,這里的test取決于servlet的名字.當(dāng)DispatcherServlet載入后,它將試圖從這個(gè)文件中載入應(yīng)用上下文。
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN" "http://www.springframework.org/dtd/spring-beans.dtd">
<beans>
<!--缺省映射處理器,不需要明確聲明,但聲明后就非常清楚使用的是哪個(gè)映射處理器 --> <bean id="beanNameUrlMapping"
class="org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping">
</bean>
<!-- 這里的name屬性有兩個(gè)職責(zé),既定義Bean的名字,也定義需要這個(gè)控制器處理的URL樣式 -->
<bean name="/hello.do"
class="com.wepull.test.HelloController">
</bean>
</beans>
上面可能有人奇怪,為什么不用id屬性,而設(shè)置name屬性。這里是因?yàn)?/span>URL中含有XML id屬性非法字符——特別是斜杠(/);
(4)配置一個(gè)視圖解析器將控制器與JSP結(jié)合起來。
將解析器的配置片段加到上面的test-servlet.xml中。
<bean id="viewResolver"
class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="prefix" value="/WEB-INF/jsp/" />
<property name="suffix" value=".jsp" />
</bean>
InternalResourceViewResolver在ModelAndView返回的視圖名前加上prefix屬性配置的前綴,再在最后加上suffix屬性配置的后綴。由于HelloController返回的ModelAndView中視圖名為welcome,所以InternalResourceViewResolver將在/WEB-INF/jsp/welcome.jsp處查找視圖。
(5)編寫呈現(xiàn)給用戶的jsp文件。
/WEB-INF/jsp/welcome.jsp
<%@ page contentType="text/html; charset=UTF-8"%>
<%@ page isELIgnored="false" %>
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
<title>Hello World!</title>
</head>
<body>
<h2>
${hello}
</h2>
</body>
</html>
完成后啟動(dòng)服務(wù)器,在瀏覽器地址欄上輸入http://locahost:8080/projectName/hello.do 就可以訪問了。
對照一下,我們發(fā)現(xiàn)SpringMVC跟Struts大同小異。只是有兩處映射,SpringMVC做的相對靈活。哪兩處呢?
(1) url和動(dòng)作(后端控制器)的映射。
springMVC里有個(gè)映射處理器(HandlerMapping)的概念。它實(shí)際上是一個(gè)處理器映射Bean,用來將一個(gè)控制器指定到一個(gè)URL上。Spring提供了三種有用的HandlerMapping的實(shí)現(xiàn):
——BeanNameUrlHandlerMapping
根據(jù)控制器的名字將控制器映射到URL
——SimpleUrlHandlerMapping
用上下文配置文件中定義的屬性集合將控制器映射到URL
——CommonsPathMapHandlerMapping
使用控制器代碼中的元數(shù)據(jù)將控制器映射到URL
(2) 邏輯視圖名和視圖對象的映射。
springMVC里還有個(gè)視圖解析器(ViewResolver)的概念。它決定了ModelAndView對象的邏輯視圖名如何解析成一個(gè)用于將結(jié)果渲染給用戶的視圖Bean..Spring有四種ViewResolver實(shí)現(xiàn):
——InternalResourceViewResolver
將邏輯視圖名解析成一個(gè)用模板文件(如JSP和Velocity模板)渲染的視圖對象
——BeanNameViewResolver
將邏輯視圖名解析成一個(gè)DispatcherServlet應(yīng)用上下文中的視圖Bean
——ResourceBundleViewResolver
將邏輯視圖名解析成一個(gè)ResourceBundler中的視圖對象
—— XmlViewResolver
從一個(gè)XML文件中解析視圖Bean,這個(gè)文件是從DispatcherServlet應(yīng)用上下文中分離出來的。
好啦,今天就算是SpringMVC的一個(gè)入門。建議大家將Struts和SpringMVC對照著學(xué)習(xí),那樣效果會(huì)更好。與人分享,集思廣益。希望大家都能夠?qū)W習(xí)快樂,工作順心。下次再見。
出處:http://blog.csdn.net/lenotang/article/details/2562348