SpringMVC运行原理
第一节 几个重要组件
1.HandlerMapping
代表请求地址到handler之间的映射。
2.HandlerExecutionChain
handler的执行链对象,由handler对象和所有handler拦截器组成。SpringMVC调用HandlerMapping接口中定义的getHandler()方法获取该对象。
3.HandlerAdapter
执行请求参数注入、类型转换、数据验证等具体操作。
第二节 关键节点
1.获取HandlerExecutionChain对象
所在API:org.springframework.web.servlet.DispatcherServlet 源码位置:1101行、916行 Tips:如果当前请求没有经过映射,那么mappedHandler是否为null呢? ①如果配置了mvc:default-servlet-handler则不为null ②如果没有配置mvc:default-servlet-handler则为null
2.获取HandlerAdapter对象
所在API:org.springframework.web.servlet.DispatcherServlet 源码位置:923行
3.调用拦截器的preHandle()方法
所在API:org.springframework.web.servlet.DispatcherServlet 源码位置:939行
4.为模型对象注入请求参数
所在API:org.springframework.web.bind.annotation.support.HandlerMethodInvoker 源码位置:170行、373行
5.调用目标handler方法
所在API:org.springframework.web.servlet.DispatcherServlet 源码位置:945行
6.调用拦截器的postHandle()方法
所在API:org.springframework.web.servlet.DispatcherServlet 源码位置:954行
7.处理视图转发相关
所在API:org.springframework.web.servlet.DispatcherServlet 源码位置:959行
8.处理异常
所在API:org.springframework.web.servlet.DispatcherServlet 源码位置:998行
9.渲染视图
所在API:org.springframework.web.servlet.DispatcherServlet 源码位置:1012行
①解析视图名称,将逻辑视图转换为物理视图
所在API:org.springframework.web.servlet.DispatcherServlet 源码位置:1204行、1266行
②渲染视图
所在API:org.springframework.web.servlet.DispatcherServlet 源码位置:1225行
③将模型数据暴露到请求域
所在API:org.springframework.web.servlet.view.AbstractView 源码位置:266行
④将模型数据保存到请求域
所在API:org.springframework.web.servlet.view.AbstractView 源码位置:374行
⑤转发
所在API:org.springframework.web.servlet.view.InternalResourceView 源码位置:209行
10.调用拦截器的afterCompletion方法
所在API:org.springframework.web.servlet.DispatcherServlet 源码位置:1030行
第三节 annotation相关
我们在前面的操作中发现,使用了mvc:default-servlet-handler和mvc:view-controller后必须使用mvc:annotation-driven。那么这是为什么呢?关键原因是他们加载使用的HandlerMapping不同。
1.三个都没有使用时有效的HandlerMapping
org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping
其中DefaultAnnotationHandlerMapping负责把所有handler类中的handler方法收集起来。
2.增加了mvc:default-servlet-handler或mvc:view-controller后有效的HandlerMapping
org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping org.springframework.web.servlet.handler.SimpleUrlHandlerMapping
很明显,DefaultAnnotationHandlerMapping没了,而SimpleUrlHandlerMapping只能映射静态资源。所以我们通过@RequestMapping映射的handler方法无效了。
3.再增加了mvc:annotation-driven后
org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping org.springframework.web.servlet.handler.SimpleUrlHandlerMapping
加入了mvc:annotation-driven后最关键的是增加了RequestMappingHandlerMapping,从而可以映射我们的handler方法。
案例:
index.jsp:
<a href="${pageContext.request.contextPath }/test/work/flow">Test work flow</a>
web.xml
<?xml version="1.0" encoding="UTF-8"?> <web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://java.sun.com/xml/ns/javaee" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" id="WebApp_ID" version="2.5"> <!-- The front controller of this Spring Web application, responsible for handling all application requests --> <servlet> <servlet-name>springDispatcherServlet</servlet-name> <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class> <init-param> <param-name>contextConfigLocation</param-name> <param-value>classpath:spring-mvc.xml</param-value> </init-param> <load-on-startup>1</load-on-startup> </servlet> <!-- Map all requests to the DispatcherServlet for handling --> <servlet-mapping> <servlet-name>springDispatcherServlet</servlet-name> <url-pattern>/</url-pattern> </servlet-mapping> </web-app>
spring-mvc.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:context="http://www.springframework.org/schema/context" xmlns:mvc="http://www.springframework.org/schema/mvc" xsi:schemaLocation="http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-4.0.xsd 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-4.0.xsd"> <context:component-scan base-package="com.atguigu.spring.mvc.handlers"/> <bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver"> <property name="prefix" value="/WEB-INF/page/"/> <property name="suffix" value=".jsp"/> </bean> <mvc:annotation-driven/> <mvc:interceptors> <mvc:interceptor> <mvc:mapping path="/**"/> <bean class="com.atguigu.spring.mvc.interceptor.WorkFlowInterceptor"/> </mvc:interceptor> </mvc:interceptors> </beans>
interceptor
import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import org.springframework.web.servlet.HandlerInterceptor; import org.springframework.web.servlet.ModelAndView; public class WorkFlowInterceptor implements HandlerInterceptor { @Override public void afterCompletion(HttpServletRequest arg0, HttpServletResponse arg1, Object arg2, Exception arg3) throws Exception { System.out.println("my after completion"); } @Override public void postHandle(HttpServletRequest arg0, HttpServletResponse arg1, Object arg2, ModelAndView arg3) throws Exception { System.out.println("my post handle"); } @Override public boolean preHandle(HttpServletRequest arg0, HttpServletResponse arg1, Object arg2) throws Exception { System.out.println("my pre handle"); return true; } }
handlers
import org.springframework.stereotype.Controller; import org.springframework.ui.Model; import org.springframework.web.bind.annotation.RequestMapping; @Controller public class WorkFlowHandler { @RequestMapping("/test/work/flow") public String myHandle(Model model) { System.err.println("my handler method"); model.addAttribute("message", "i love you!!!"); return "target"; } }
结果:
信息: Server startup in 33416 ms my pre handle my handler method my post handle my after completion
页面结果:
Target i love you!!!