SpringMVC 工作原理(书面解释)
- 客户端请求提交到DispatcherServlet;
- 由DispatcherServlet控制器寻找一个或者多个HandlerMappring(处理器映射),找到处理请求的Controller;
- DispatcherServlet将请求提交到Controller;
- Controller调用业务逻辑处理之后,返回ModelAndView;
- DispatcherServlet寻找一个或多个ViewResolver(视图解析器),找到ModelAndView指定的视图;
- 视图负责将结果显示到客户端。
下面,通过最简单的、没有信息传递的页面跳转案例,对SpringMVC运行的基本流程解读,对上面的流程描述背后的程序逻辑进行复现。
- 本次案例的SpringMVC工程目录结构图
使用SpringMVC的工程目录上显示,与一般的web项目比较,多出来一个springmvc-servlet.xml的配置文件。
现在,从这个项目的index界面进入并进行操作,从而伴随查看程序的运行过程。
- index.jsp的body内容如下:
<body>
没注册的用户,请<a href="register">注册</a> ! <br>
已注册的用户,去<a href="login">登录</a> !
</body>
网页的功能非常简单,就是有两个跳转链接,点击不同的链接可以跳转到不同的页面,显示不同的页面内容。
按照Java动态web项目运行的规则,当页面发生请求的时候,web.xml会最先感知到请求,并根据请求的类型对请求要产生的下一步操作进行引导。所以,有必要看一下web.xml文件的内容。
- web.xml
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://xmlns.jcp.org/xml/ns/javaee" xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd" id="WebApp_ID" version="4.0">
<display-name>FirstSpringMVC</display-name>
<welcome-file-list>
<welcome-file>index.html</welcome-file>
<welcome-file>index.htm</welcome-file>
<welcome-file>index.jsp</welcome-file>
<welcome-file>default.html</welcome-file>
<welcome-file>default.htm</welcome-file>
<welcome-file>default.jsp</welcome-file>
</welcome-file-list>
<!-- 部署 DispatcherServlet -->
<servlet>
<servlet-name>springmvc</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>springmvc</servlet-name>
<!-- 处理所有 URL -->
<url-pattern>/</url-pattern>
</servlet-mapping>
</web-app>
根据配置文件的配置信息:
请求被<url-pattern>/</url-pattern>截获,随后,根据<servlet-name>springmvc</servlet-name>找到了对应的实体类org.springframework.web.servlet.DispatcherServlet,DispatcherServlet将请求进行分发,分发的根据是springmvc-servlet.xml的配置信息。(之所以web.xml能够找到springmvc-servlet.xml,是因为servlet-name标签的内容是springmvc)
- springmvc-servlet.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"
xmlns:mvc="http://www.springframework.org/schema/mvc"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/mvc
http://www.springframework.org/schema/spring-mvc.xsd">
<bean name="/login" class="controller.LoginController" />
<bean name="/register" class="controller.RegisterController" />
</beans>
配置信息显示:如果请求是login,就定位到controller.LoginController这个控制器;如果请求是register,就定位到controller.RegisterController;
由于是相同的业务逻辑,所以对其中的一个进行跟踪。以login为例,LoginController控制器类的内容如下:
- LoginController.java
package controller;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.springframework.web.servlet.ModelAndView;
import org.springframework.web.servlet.mvc.Controller;
public class LoginController implements Controller {
@Override
public ModelAndView handleRequest(HttpServletRequest arg0, HttpServletResponse arg1) throws Exception {
return new ModelAndView("/WEB-INF/jsp/login.jsp");
}
}
LoginController类实现了Controller接口,并实现了其handleRequest()方法。这个方法的返回值是一个视图,也可以说是一个页面,这个页面就是/WEB-INF/jsp/login.jsp。
到这里,SpringMVC就完成了一次请求响应。但是,前面的描述中,还提到了一个东西叫视图解析器(ViewResolver)。视图解析器的工作原理如下:
同样是在springmvc-servlet.xml这个配置文件中,还有一块配置内容:
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver" id="internalResourceViewResolver">
<!-- 前缀 -->
<property name="prefix" value="/WEB-INF/jsp/"></property>
<!-- 后缀 -->
<property name="suffix" value=".jsp"></property>
</bean>
通过这一块配置内容,对于每一个页面路径,都可以自动添加前缀和后缀。有了视图解析器,控制器返回视图的时候,就可以只写一个文件名。(比如,有了视图解析器,RegisterController可以是这个样子的)
- RegisterController.java
package controller;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.springframework.web.servlet.ModelAndView;
import org.springframework.web.servlet.mvc.Controller;
public class RegisterController implements Controller {
@Override
public ModelAndView handleRequest(HttpServletRequest arg0, HttpServletResponse arg1) throws Exception {
return new ModelAndView("register");
}
}
通过前面配置的视图解析器,可以将RegisterController类返回的"register"封装为"/WEB-INF/jsp/register.jsp".