首页 > 其他分享 >day15-SpringMVC执行流程

day15-SpringMVC执行流程

时间:2023-02-20 22:47:50浏览次数:58  
标签:SpringMVC 流程 request 点击 视图 day15 step ModelAndView 方法

SpringMVC执行流程

1.SpringMVC执行流程分析图

例子

(1)创建 HaloHandler

package com.li.web.debug;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.servlet.ModelAndView;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

/**
 * @author 李
 * @version 1.0
 */
@Controller
public class HaloHandler {
    //编写方法,响应请求,返回一个 ModelAndView对象
    @RequestMapping(value = "/debug/springmvc")
    public ModelAndView halo(HttpServletRequest request, HttpServletResponse response) {
        ModelAndView modelAndView = new ModelAndView();
        //对应到 WEB-INF/pages/ok.jsp (ok的前后缀是你在视图解析器中配置的前后缀)
        modelAndView.setViewName("ok");
        //在model中放入数据 k-v,ModelAndView的属性也会被springmvc放入到request域中
        modelAndView.addObject("name", "齐天大圣");
        return modelAndView;
    }
}

(2)创建ok.jsp,作为响应后跳转的页面

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>ok</title>
</head>
<body>
<h1>进入到ok页面</h1>
<h2>name-${requestScope.name}</h2>
</body>
</html>

2.源码debug

我们以上述代码为例子进行源码分析。

(1)在 DispatcherServlet 的 doService 方法中打上断点,在浏览器中访问目标方法,可以看到光标跳转到断点处:当请求发送到服务器时,tomcat 将 http 请求包装成 request 对象,前端控制器 DispatcherServlet 使用 doService() 方法接收这个 request 对象。

image-20230220213910330 image-20230220203713801

(2)点击 step over 前进,可以看到这里的 getWebApplicationContext() 就是 spring 容器对象。也就是说,当接收到url请求的时候,前端控制器就会进行spring 容器的初始化,将各种 bean 放入到容器中。

image-20230220204141473

(3)点击 step over,在经过一系列处理之后调用了 doDispatch() 方法。这是一个核心的方法。

image-20230220204445920

(4)点击 step into,进入 doDispatch() 方法,该方法中首先定义了一个处理器执行链,它用于存放拦截器(多个)和目标 Handler。然后定义了ModelAndView 对象,用于存放视图信息和数据。

image-20230220213959367 image-20230220205326737

执行下面的语句后,就通过映射拿到了处理器执行链 HandlerExecutionChain 的目标 Handler 和拦截器链

image-20230220210019024

(5)点击 step over,跳到如下:获取适配器,包含要执行的目标 Handler

image-20230220210743139 image-20230220214118074

(6)点击 step over,跳到如下:调用 handle() 方法,进行反射调用目标 Handler。

image-20230220211201477

(7)点击 step into,进入到 handle 方法中调用的 handleInternal() 方法,handleInternal() 方法中反射调用了目标 Handler的方法,然后返回视图对象。

image-20230220211812556 image-20230220214704657

(8)在目标方法中打上断点,点击 resume 进入目标方法

image-20230220212955654

(9)点击 step over,ModelAndView 对象返回给适配器。

image-20230220213514030 image-20230220214807691

(10)然后返回到第7步的方法中,因此第7步中返回的视图就是目标方法操作后返回的视图对象

image-20230220215320257

(11)继续一直点击 step over,方法 return 返回到第6步,将获取的视图对象返回给前端控制器。

image-20230220215703488 image-20230220215805035

(12)点击 step over,在前端控制器的 doService 方法中执行如下语句,processDispatchResult() 方法对前面返回的视图进行解析。

image-20230220220003548

(13)step into 进入processDispatchResult 方法,该方法调用 render() 进行渲染。

image-20230220220233835

(14)render() 方法从 ModelAndView 对象中得到视图名称 viewName,如果 viewName 不为空,就进行视图解析。

image-20230220220630961

resolveViewName() 方法进行视图解析,然后返回 view 给前端控制器。

@Nullable
protected View resolveViewName(String viewName, @Nullable Map<String, Object> model,
      Locale locale, HttpServletRequest request) throws Exception {

   if (this.viewResolvers != null) {
      for (ViewResolver viewResolver : this.viewResolvers) {
         View view = viewResolver.resolveViewName(viewName, locale);
         if (view != null) {
            return view;
         }
      }
   }
   return null;
}
image-20230220220937160

(15)返回前端控制器之后,又调用 View的 render() 方法进行视图的渲染。

注意和前端控制器的 render 方法区分

image-20230220221538252 image-20230220222131962

View 的 render() 方法:

@Override
public void render(@Nullable Map<String, ?> model, HttpServletRequest request,
      HttpServletResponse response) throws Exception {

   if (logger.isDebugEnabled()) {
      logger.debug("View " + formatViewName() +
            ", model " + (model != null ? model : Collections.emptyMap()) +
            (this.staticAttributes.isEmpty() ? "" : ", static attributes " + this.staticAttributes));
   }

   Map<String, Object> mergedModel = createMergedOutputModel(model, request, response);
   prepareResponse(request, response);
    //渲染合并输出模型
   renderMergedOutputModel(mergedModel, getRequestToExpose(request), response);
}

renderMergedOutputModel() 方法通过请求转发跳转到相应页面:

image-20230220221857739 image-20230220222033914

(16)最后 tomcat 返回 http响应,浏览器显示页面

image-20230220222507677

3.练习

  1. 将之前的 SpringMVC 异常处理相关代码和案例写一遍
  2. 简述原生的 SpringMVC 执行流程,并画出示意图
  3. debug SpringMVC 的执行流程源码,加深理解

标签:SpringMVC,流程,request,点击,视图,day15,step,ModelAndView,方法
From: https://www.cnblogs.com/liyuelian/p/17139256.html

相关文章

  • 客户端跟redis的一次通信流程
    1.在redis启动初始化的时候,redis会将连接应答处理器跟AE_READABLE事件关联起来,接着如果一个客户端跟redis发起连接,此时会产生一个AE_READABLE事件,然后由连接应答处理......
  • activiti 6 遍历流程任务
    XMLInputFactoryxif=XMLInputFactory.newInstance();List<ProcessDefinition>list=repositoryService.createProcessDefinitionQuery().latestVersi......
  • app上架需要准备什么以及上架流程
     上架前需要准备的材料:众所周知,应用市场主要分为两类,一类是AppStore,另一类是安卓市场。而安卓市场又分为:第三方市场(如:应用宝、360手机助手、豌豆荚),和手机厂商市......
  • springmvc异常处理
    1 DispatcherServlet的handlerExceptionResolvers在DispatcherServlet初始化过程中,执行如下初始化方法protectedvoidinitStrategies(ApplicationContextcontext......
  • SpringMVC上传大文件的三种解决方案
    ​ 1,项目调研因为需要研究下断点上传的问题。找了很久终于找到一个比较好的项目。 在GoogleCode上面,代码弄下来超级不方便,还是配置hosts才好,把代码重新上传到了githu......
  • 流程控制
    引子流程控制即控制流程,具体指控制程序的执行流程,而程序的执行流程分为三种结构:顺序结构(之前我们写的代码都是顺序结构)、分支结构(用到if判断)、循环结构(用到while与for)一......
  • 完整商业软件的开发流程
    完整商业软件的开发流程1、产品立项,确定要做的内容、效果和收益2、开始产品原型,确定初步逻辑和技术——>需求评审,定开发节点、UI节点3、根据原型出设计图,确定实现细节......
  • Java流程控制
    Java流程控制Scanner对象next():一定要读取到有效字符后才可以结束输入对输入有效字符之前遇到的空白,next()方法会自动将其去掉只有输入有效字符后才能将其后面的......
  • springMVC中的ModelMap作用及用法
    概念ModelMap的作用:ModelMap对象主要用于传递控制方法传递数据到结果页面。类似于request的setAttribute方法的作用。所以我们要想在jsp页面获取数据,只要将数据放到ModelMa......
  • 项目一众筹网02_4_SpringMVC访问静态资源:/*,/,和*.do和*.html的区别
    系列文章目录文章目录​​系列文章目录​​​​项目答疑​​​​这样做必有它的理由,但是却不是十全十美的,得必有舍​​项目答疑我们这里为什么@RequestMapping(“admin/do/......