zoukankan      html  css  js  c++  java
  • Spring MVC框架:第十二章:运行原理

    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!!! 
    
  • 相关阅读:
    go语言学习之从例子开始
    分享一个不错的Unittest测试报告
    selenium各种定位方法(转)
    HTMLTESTRunner自动化测试报告增加截图功能
    selenium之chrome驱动版本
    Python基础(六) python生成xml测试报告
    vue+elementUI 表头按钮
    vue+elementUI滚动条
    vue表格表头添加按钮
    elementUI升级版本后Dialog弹空不显示问题
  • 原文地址:https://www.cnblogs.com/javawxid/p/12812064.html
Copyright © 2011-2022 走看看