通过Debug探索SpringMVC执行过程

news2025/7/21 5:42:33

文章目录

  • 对SpringMVC的理解
  • SpringMVC工作原理
  • SpringMVC工作流程

对SpringMVC的理解

MVC:MVC是一种设计模式
MVC的原理图:
在这里插入图片描述

  • M-Model 模型(完成业务逻辑:有javaBean构成,service+dao+entity)
  • V-View 视图(做界面的展示 jsp,html……)
  • C-Controller 控制器(接收请求—>调用模型—>根据结果派发页面)

SpringMVC工作原理

springMVC是一个MVC的开源框架,springMVC=struts2+spring,springMVC就相当于是Struts2加上sring的整合,但是这里有一个疑惑就是,springMVC和spring是什么样的关系呢?这个在百度百科上有一个很好的解释:意思是说,springMVC是spring的一个后续产品,其实就是spring在原
有基础上,又提供了web应用的MVC模块,可以简单的把springMVC理解为是spring的一个模块(类似AOP,IOC这样的模块),网络上经常会说springMVC和spring无缝集成,其实springMVC就是spring的一个子模块,所以根本不需要同spring进行整合。

在这里插入图片描述

  • 用户发送请求至前端控制器DispatcherServlet。
  • DispatcherServlet收到请求调用HandlerMapping处理器映射器。
  • 处理器映射器找到具体的处理器(可以根据xml配置、注解进行查找),生成处理器对象及处理器拦截器(如果有则生成)一并返回给DispatcherServlet。
  • DispatcherServlet调用HandlerAdapter处理器适配器。
  • HandlerAdapter经过适配调用具体的处理器(Controller,也叫后端控制器)。
  • Controller执行完成返回ModelAndView。
  • HandlerAdapter将controller执行结果ModelAndView返回给DispatcherServlet。
  • DispatcherServlet将ModelAndView传给ViewReslover视图解析器。
  • ViewReslover解析后返回具体View。
  • DispatcherServlet根据View进行渲染视图(即将模型数据填充至视图中)。
  • DispatcherServlet响应用户。

在这里插入图片描述

相关组件介绍

  • 前端控制器DispatcherServlet:其作用就是接收请求,响应结果,相当于转发器,中央处理器。有了dispatcherServlet减少了其它组件之间的耦合度。 用户请求到达前端控制器,它就相当于mvc模式中的c,dispatcherServlet是整个流程控制的中心,由它调用其它组件处理用户的请求,dispatcherServlet的存在降低了组件之间的耦合性。
  • 处理器映射器HandlerMapping:根据请求的url查找Handler HandlerMapping负责根据用户请求找到Handler即处理器,springmvc提供了不同的映射器实现不同的映射方式,例如:配置文件方式,实现接口方式,注解方式等。
  • 处理器适配器HandlerAdapter:按照特定规则(HandlerAdapter要求的规则)去执行Handler 通过HandlerAdapter对处理器进行执行,这是适配器模式的应用,通过扩展适配器可以对更多类型的处理器进行执行。
  • 处理器Handler:Handler 是继DispatcherServlet前端控制器的后端控制器,在DispatcherServlet的控制下Handler对具体的用户请求进行处理。 由于Handler涉及到具体的用户业务请求,所以一般情况需要工程师根据业务需求开发Handler。
  • 视图解析器View resolver:进行视图解析,根据逻辑视图名解析成真正的视图(view) View Resolver负责将处理结果生成View视图,View Resolver首先根据逻辑视图名解析成物理视图名即具体的页面地址,再生成View视图对象,最后对View进行渲染将处理结果通过页面展示给用户。 springmvc框架提供了很多的View视图类型,包括:jstlView、freemarkerView、pdfView等。 一般情况下需要通过页面标签或页面模版技术将模型数据通过页面展示给用户,需要由工程师根据业务需求开发具体的页面。
  • 视图View: View是一个接口,实现类支持不同的View类型(jsp、
    freemarker、pdf…),需要工程师开发

SpringMVC工作流程

接下来我们借助IDEA的Debug功能探索SpringMVC工作流程。首先我们要确认我们的断点打在哪里。

在这里插入图片描述

如果是打在这里的话,那么相当于直接到了处理器那一步,前面的步骤我们都看不到了。根据上面的原理图我们可以知道第一步肯定与DispatcherServlet有关,根据后缀然后再查看它的结构我们可以知道:DispatcherServlet继承自HttpServlet,其本质就是一个Servlet。
在这里插入图片描述
根据Servlet的生命周期我们可以知道:每次请求Servlet时,Servlet容器都会调用Servlet的service()方法对请求进行处理。

如果还不是很了解Servlet的生命周期,可以看看我的另一篇文章:
Servlet初识(执行流程、生命周期)

所以我们找到service方法然后打上断点,发现卡上了,说明入口是对的:
在这里插入图片描述

接下来我们看重点:
在这里插入图片描述
我们从字面意思可以知道这里是对请求进行一个分发,我们步入这个方法,这时候我们可以看到一个HandlerExecutionChain,这就与前面的原理图完全吻合。
在这里插入图片描述

此时我们只需要寻找后面是谁给这个mappedHandler赋值的:
在这里插入图片描述
我们可以发现通过传入一个被包装过的request请求就得到了我们的HandlerExecutionChain处理器执行链。我们计算这个表达式看一下他的执行结果:
在这里插入图片描述

我们可以看到要执行的过滤器以及处理器handler。其中他还拿到了我们的method,也就是说底层是使用方法反射然后使用invoke执行。

注:在IDEA中右键语句,点击计算表达式;
在这里插入图片描述
注意:只有断点运行到了这里才能计算

综上所属HandlerExecutionChain处理器执行链就是一个类的实例,其中装着处理器以及拦截器,其中处理器(handler)就是你写的Controller,拦截器有默认的和你自己定义的。

然后我们步入这个getHandler方法看看,根据原理图这个方法肯定和HandlerMapping处理器映射器有关:
在这里插入图片描述
我们看到这个handlerMappings有三个处理器映射器:
在这里插入图片描述
也就是是说这里遍历依次使用每个处理器映射器,看哪个能拿到处理器链,然后对这个处理器链进行返回。

这里我们依次步过,发现RequestMappingHandlerMapping可以拿到处理器链。这里我们继续进入mapping.getHandler(request)方法中查看:
在这里插入图片描述
我们发现getHandlerInternal方法就可以拿到我们的处理器,那么我们就再进入getHandlerInternal方法:

注意这里进入的是AbstractUrlHandlerMapping中的getHandlerInternal方法

在这里插入图片描述
发现从lookupHandler方法拿到处理器,我们继续进入:

在这里插入图片描述
这里我们就可以看到:通过urlPath(也就是我们设置的路径)在map中去拿到处理器handler

至此,验证了SpringMVC的执行流程:前端控制器接收到客户端的请求后,通过处理器映射器handlerMappings,根据路径urlPath去匹配选择处理器handler,最终返回一个处理器执行链对象HandlerExcutionChain。


接下来我们继续查看doDispatch方法,在得到了mappedHandler这个处理器链之后,它再通过getHandlerAdapter方法拿到HandlerAdapter处理器适配器。
在这里插入图片描述
然后后面调用了这个HandlerAdapter,得到了mv,这个mv就是ModelAndView
在这里插入图片描述
我们看到这里有一个mappedHandler.getHandler(),也就是通过处理器链得到处理器。我们进入getHandler方法看看:

在这里插入图片描述
我们发现就是把处理器链的处理器给拿出来,这个处理器与我们前面的Controller也吻合:
在这里插入图片描述
那么接下来我们就看看ha.handle方法,我们进入这个方法;
在这里插入图片描述

继续进入,来到了RequestMappingHandlerAdapter(也就是我们当前获得的处理器适配器)中的handleInternal方法。
在这里插入图片描述
一眼就可以看到invokeHandlerMethod方法,也就是说这个方法用来执行处理器中对应的方法。我们进入这个方法:
在这里插入图片描述
这都不重要我们继续往下看,可以看到这个核心方法;
在这里插入图片描述
这个地方才真正的开始调用并执行处理器的方法。我们可以初步判断就是这个方法内部拿到我们的请求,然后解析我们的请求参数,再然后传入我们的处理器方法,最后去执行。

我们继续进入这个方法:

在这里插入图片描述
还没有看到解析请求参数的步骤继续步入invokeForRequest方法:

在这里插入图片描述
也就是说getMethodArgumentValues方法中应该就是参数的解析过程,我们继续进入:
在这里插入图片描述

我们可以看看这里的resolvers参数解析器:
在这里插入图片描述
我们可以看到几种常用的参数解析器:
在这里插入图片描述

就这样循环解析每一个参数,得到参数值列表args:
在这里插入图片描述
在这里插入图片描述
最后传入doInvoke方法

拓:
考虑到可能有频繁多次的访问,为了避免重复的调用同一个解析器影响效率,在resolveArgument方法的内部对解析器建立了缓存:
在这里插入图片描述
在这里插入图片描述
也就是说第一次会遍历解析器看哪个适配,如果有适配的就会根据参数把他存入缓存之中再去解析参数。第二次就会根据参数在缓存中找到解析器,直接拿来解析使用,不需要再去遍历看是否适配,如此更加高效。
在这里插入图片描述
我们可以继续看看resolver.supportsParameter(parameter)方法,它是用来确认是否适配解析器的:
在这里插入图片描述
大致就是通过注解判断使用什么解析器。

至此,验证了SpringMVC的执行流程:前端控制器通过处理器适配器HandlerAdapter,调用处理器的方法(也就是Controller中的方法),然后处理器执行后返回模型视图对象ModelAndView给处理器适配器,适配器再将模型视图对象返回给前端控制器.

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/395224.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

苹果或3月8日开新品发布会是真的吗?

苹果或3月8日开新品发布会是真的吗?都发布什么产品?iPhone14即将推出至少1种新颜色 #iPhone #苹果 苹果怎么又出小屏手机? S1 多好用,单手就能操作。 诶,苹果这么小字都看不清楚,可是这是有 H5 又怎么样…

张云雷再唱《探清水河》,柠檬们热情似火,预约通道都要崩溃了

起德云社的相声演员,很多都是唱歌的高手,比如说“德云一哥”岳云鹏,当然更少不了小辫张云雷。张云雷的成功之处在于,他总能够化腐朽为神奇,让不可能成为可能,这是其他相声演员所不能比拟的。 就拿他的成名曲…

《安富莱嵌入式周报》第305期:超级震撼数码管瀑布,使用OpenAI生成单片机游戏代码的可玩性,120通道逻辑分析仪,复古电子设计,各种运动轨迹函数源码实现

往期周报汇总地址:嵌入式周报 - uCOS & uCGUI & emWin & embOS & TouchGFX & ThreadX - 硬汉嵌入式论坛 - Powered by Discuz! 说明: 谢谢大家的关注,继续为大家盘点上周精彩内容。 视频版: https://www.bi…

Redis锁与幂等性不得不说的故事

前言: 相信很多小伙伴对缓存锁都不陌生,但是简单的缓存锁想要用好还是需要一些功力。本文总结了笔者多年使用缓存所的一些心得,欢迎交流探讨~ 幂等模型: 幂等场景一般由查重写入两步操作组成,两步操作组成一个最小完…

SpringSecurity支持WebAuthn认证

WebAuthn是无密码身份验证技术,解决了密码泄露的风险,主流的浏览器都支持。有很多开源的类库实现了WebAuthn规范,Java下流行的类库有:webauthn4jjava-webauthn-serververtx-authSpring Security官方暂时未支持WebAuthn&#xff0c…

怎样编写java程序

搭建好了Java开发环境之后,下面就来学习一下如何开发Java程序。为了让初学者更好地完成第一个Java程序,接下来通过几个步骤进行逐一讲解。 1.编写Java源文件 在D盘根目录下新建一个test文件夹,并在该文件夹中新建文本文档&#…

运维自动化之Ansible

一:Ansible的概述ansible的简介Ansible是一款为类Unix系统开发的自由开源的配置和自动化工具。它用Python写成,类似于saltstack和Puppet,但是有一个不同和优点是我们不需要在节点中安装任何客户端。它使用SSH来和节点进行通信。Ansible基于 P…

【C++】Clang-Format:代码自动格式化(看这一篇就够了)

文章目录Clang-format格式化C代码1.引言&安装1.1引言1.2 安装2. 配置字解释2.1 language 编程语言2.2 BaseOnStyle 基础风格2.3 AccessModifierOffset 访问性修饰符偏移2.4 AlignAfterOpenBracket 开括号后的对齐2.5 AlignArrayOfStructures 对齐结构体数组2.6 AlignConsec…

中电金信Gien享汇・大数据专题|金融行业数据架构及模型演进

本期嘉宾 陈子刚 中电金信商业分析事业部华南区总经理 毕业于复旦大学,硕士研究生;拥有16年以上金融行业商业智能领域从业经验;曾就职于中国工商银行、Teradata、东亚银行,服务于平安银行、广发银行、招商银行、广东农信、招商…

IT项目经理的自我修养手册(续)

上一节从工作职责和如何提升IT项目经理的个人能力讲述了作为一个职业项目经理的自我修养, 入口地址: IT项目经理的自我修养手册 本节将会从计划、组织、领导、控制这四个方面,系统的介绍作为职业项目经理的工作职责和工作范围。 一、计划 …

Springboot怎么实现WebSocket通信(一)

文章示例环境配置信息jdk版本:1.8开发工具:Intellij iDEA 2020.1springboot:2.3.9.RELEASE什么是WebSocket?WebSocket,是HTML5下一种新的协议,支持web浏览器和服务器端之间双向全双工通信 ,基于TCP协议实现。WebSocket主要特性1、…

console使用方法介绍

console是在写前端Javascript时经常会使用到,我平时使用最多的是console.log,相比大多数人也是如此吧! 下面一起来看一下强大的console吧! 01函数(属性) 包含如下函数 / 属性:memory、assert、c…

2023年最新最全 VSCode 插件推荐

Visual Studio Code 是由微软开发的一款免费的、针对于编写现代Web和云应用的跨平台源代码编辑器。它包含了一个丰富的插件市场,提供了很多实用的插件。下面就来分享 2023 年前端必备的 VS Code 插件! 前端框架 ES7 React/Redux/React-Native snippets …

洗地机哪个牌子比较好?家用洗地机推荐

洗地机是可以做到扫地的同时把地也拖了,可以做到高效清洁,减轻家务负担。还具有杀菌功能,不用担心拖完地之后地面还有细菌滋生啦,洗地机可以做到即洗即干,无需等待拖完地之后还潮湿不能走路,尤其对于木地板…

简述下 React 的事件代理机制?

React 并不会把所有的处理函数直接绑定在真实的节点上。而是把所有的事件绑定到结构的最外层,使用一个统一的事件监听器,这个事件监听器上维持了一个映射来保存所有组件内部的事件监听和处理函数。 当组件挂载或卸载时,只是在这个统一的事件…

LIN资料整理(一):LIN入门简介

目录 一、LIN是什么 1、LIN简介 2、LIN的适用场景 3、LIN协议版本 4、LIN的主从机节点、主从机任务与LIN总线特点 一、LIN是什么 1、LIN简介 LIN 是Local Interconnect Network 的缩写,是基于UART/SCI(Universal Asynchronous Receiver-Transmitter / Serial…

2023级社科院与杜兰大学金融管理硕士还有少量名额,期待与你共谱人生新篇章

我们都是平凡人,过着平凡的生活,但若能有所热爱有所坚持,便会发出自己的光芒。就像我们在社科院与杜兰大学金融管理硕士项目读研,通过更加系统的学习,不仅丰富了我们的知识,还提升了我们的思维,…

NB-IOT宣传这么多年,这次总算用好了吧

一、方案概述随着实体经济快速发展,石化、港口、货场、工地等区域规模日益扩大,厂区面积广阔、环境复杂、作业人员和车辆众多,如无法实时掌握工作人员状态及外来人员位置、外来车辆情况等问题,将存在非常大的安全隐患。今天小编介…

ChatGPT在工业领域的用法

在工业数字化时代,我们需要怎么样的ChatGPT? 近日,ChatGPT热度高居不下,强大的人机交互能力令人咋舌,在国内更是掀起一股讨论热潮。一时间,这场由ChatGPT引起的科技飓风,使得全球最顶尖科技力量…

notepad++学习小技巧

不要小瞧了notepadd 这个可是我们的cv好帮手。。。 实战1背景,我找一个同事要表结构 结果他给我了一个xml。顿时一懵,我也不知道为啥好像是从前端扣下来的。 建表我只需要 columnName, displayName当作是comment, dataTypeNamecolumnType借鉴…