Spring mvc 是Spring 框架最重要的模块之一。它构建于强大的Spring IOC 容器之上,大量使用容器的特性就行简化。
M model 模式----封装视图展示的应用数据
V view 视图-----只是显示数据,不包含任何业务逻辑
C controller 控制器----接受用户请求并调用后端服务进行业务处理
工作原理:
图1.Spring MVC中请求处理主流程
Dispatcher Servlet控制器,该核心控制器需要在Java Web项目中的Web.xml文件中进行描述。
从图
1 中可以看出,其作为 MVC 框架的前端控制器,每个 Web 请求都必须通过它,以便使它能够管理整个请求处理过程。
Request
— 1
来自用户的请求被
Dispatcher Servle 拦截下来
Request
— 2.1
Dispatcher Servlet
将请求推送到 Handler Mapping 映射,在上下文环境中找到对应的 Controller (一个用 @Controller 进行修饰的 java 类 spring 3.0 )
Controller 经过
Handler Mapping 处理,请求被定位到对应的控制器类,再经过 Dispatcher Servlet 进行分发
request—2.2
请求被分发到指定的拦截器类中进行处理,这里可以调用一些业务逻辑的服务类进行特定功能的实现,而数据的传输一般采用
model 属性进行封装 model.addtribute(“propertyName”,(Object) property)
ModelAndViow
最后返回一个视图名称
viewName
View Name
Dispatcher Servlet
拦截并让发送给视图解析器进行解析
视图解析器
View Resolver ,按照优先级进行解析
View
确定下视图名称与路径
Modle
属性在
View 中进行实现与初始化
Response—1
容器拦截并准备返回
Response
— 2
mvc
框架将实现好的视图(数据注入后)页面返回给用户
上诉逻辑的实现,需要在
Java Web 项目中的 Web 描述符 web.xml (配置文件)进行部署。
关键是配置
servlet ,要使用 Spring mvc 必须配置其核心的控制器类——Dispatcher Servlet ,当然可以配置多个 DispatcherServlet 实例,以方便团队间的协作开发。如图 2 ,图 3 所示。
1.<servlet>
:
确定一个
servlet 和其实现的类,这里实现的类是 DispathcerServlet
<Sevlet-name> :
第一个作用是与 <servlet-mapping> 中 <serlvet-name> 进行关联,并根据其下的 <url-pattern> 进行请求路径的拦截;
第二个作用是让
DispatcherServlet 决定加载哪个 Spring MVC 配置文件。默认情况下,如图 3 查找的是 mannage 加上 -servlet.xml 的文件;也可以自己在 <servlet> 参数 contextConfigLocation中明确指定一个配置文件,如图 2 查找的是 product-service.xml 文件。
<url-pattern>
中 ’ / ’ 表示当前路径下的所有请求
2 product-servlet
图3 manage-servlet
2.采用<context-param>加载上下文文件
为了让 Spring
加载 court-servlet.xml 之外的文件,还必须在 web.xml 中定义 serlvet 监听器(<listener>) ContextLoaderListener 。
默认情况下,它加载配置文件 /WEB-INF/applicationContext.xml
;也可以在上下文 参数<param-name>contextConfigLocation中指定自己的文件<param-value>/WEB-INF/manage-service.xml(你可以指定多个用逗号或者空格分隔的配置文件),如图 4 所示。
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>/WEB-INF/base-service.xml ,
/WEB-INF/base2-service.xml </param-value>
</context-param>
<listener> <listener-class> org.springframework.web.context.ContextLoaderListener
</listener-class>
</listener>
图 4
context-param
ContextLoaderListener 加载指定的
Bean 配置文件到根应用上下文中,而每个DispatcherServlet 将其配置文件加载到自己的应用上下文,并且引用根应用上下文为其上级。每个 DispatcherServlet 实例加载的上下文可以访问甚至覆盖根应用上下文中声明的 Bean( 反之则不行 ) 。但是, DispatcherServlet 实例加载的上下文无法互相访问。
现在大的环境已经配置好了,剩下的就是采用 @Controller
修饰控制器,这时我们知道哪个类是控制器类,哪个类一般的 JAVA 类了,但是让 spring mvc 容器知道呢?
3. 采用
<context:component-scan> 元素启用 Spring 的组件扫描功能。
product-service.xml
首先加入如下元素,让其扫描指定包中的类
<context:component-scan
base-package="com.future.product.web" />
然后还需要加入下面 2
个声明,因而在 web 上下文中注册
DefaultAnnotationHandlerMapping和 AnnotationMethodHandlerAdapter实例,它们分别在类级别和方法级别处理
@ReqeustMapping 注解。
<bean class="org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping">
<property name="interceptors">
<list>
<bean class="com.future.product.web.CountTimeInterceptor" />
</list>
</property>
</bean>
类级别的处理,可以加入自定义的拦截器 <property>
CountTmieInerceptor.java 是实现
HandlerInterceptor 接口或者继承 HandlerInterceptorAdapter 的 java 类,可以在 Controller 类执行方法前后进行额外的操作
<bean class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter">
<property name="webBindingInitializer">
<bean class="com.future.product.web.ProductBindingInitializer"></bean>
</property>
</bean>
方法级别的处理,可以加入自定义的方法 binding
类 <property>
ProductBindingInitializer.java 是实现
WebBindingInitializer 接口,实现自定义属性的转化与编辑。
现在大体上环境已经部署好了,下面通过一个简单的例子来进行验证、体会。
Eclipse + Tomcat7.0 +Spring MVC
1.项目在Eclipse中的结构如图4所示。
4 应用层次结构
2.需要导入的包如下:
commons-logging-1.1.1.jar
commons-logging-api-1.1.jar
log4j-1.2.16.jar
org.springframework.asm-3.0.7.RELEASE.jar
org.springframework.beans-3.0.7.RELEASE.jar
org.springframework.context.support-3.0.7.RELEASE.jar
org.springframework.context-3.0.7.RELEASE.jar
org.springframework.core-3.0.7.RELEASE.jar
org.springframework.expression-3.0.7.RELEASE.jar
org.springframework.jdbc-3.0.7.RELEASE.jar
org.springframework.web.portlet-3.0.7.RELEASE.jar
org.springframework.web.servlet-3.0.7.RELEASE.jar
org.springframework.web-3.0.7.RELEASE.jar
standard-1.1.2.jar
3.Web.xml如下:
<?xml version="1.0" encoding="UTF-8"?>
<web-app version="2.5"
xmlns="http://java.sun.com/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee
http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd">
<display-name>Future Future Future </display-name>
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>
/WEB-INF/base-service.xml
</param-value>
</context-param>
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<servlet>
<servlet-name>product</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>/WEB-INF/product-service.xml</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet>
<servlet-name>manage</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>product</servlet-name>
<url-pattern>/login</url-pattern>
<url-pattern>/product/*</url-pattern>
</servlet-mapping>
<servlet-mapping>
<servlet-name>manage</servlet-name>
<url-pattern>/manage/*</url-pattern>
</servlet-mapping>
</web-app>
4.base.xml中还没有具体写东西,就先不管了,下面的是product-service.xml
<?xml version="1.0" encoding="UTF-8"?>
<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-3.0.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-3.0.xsd
http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-3.0.xsd">
<context:component-scan base-package="com.future.login.web" />
<bean class="org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping" />
<bean class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter" />
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="prefix" value="/WEB-INF/jsp/" />
<property name="suffix" value=".jsp" />
<property name="order" value="1" />
</bean>
</beans>
5.在 com.future.login.web
编写一个 Login.java
package com.future.login.web;
import javax.servlet.http.HttpServletRequest;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.SessionAttributes;
@Controller
@RequestMapping("/login")
@SessionAttributes("user")
public class Login {
@RequestMapping(method = RequestMethod.GET)
public String welcome(HttpServletRequest request){
return "welcome";
}
@RequestMapping(value = "/pr" ,method = RequestMethod.GET)
public String welcome2(HttpServletRequest request){
return "welcome";
}
}
6.在 WEB-INF/jsp/
目录下创建 Welcime.jsp
<body>
<h1> 欢迎您
2</h1>
<hr />
</body>