SpringMVC执行流程

1 流程对比

1.1 原生servlet开发流程

  1. 根据需求编写servlet程序
  2. 在web.xml 中通过配置,指定servlet所能处理的请求,即建立servlet与请求路径间的映射
  3. 在Servlet的service方法中对路径进行再判断,从而进行具体的逻辑处理
  4. servlet参数从request中获取,servlet的响应则放入response对象中
<servlet><!--自定义的servlet--><servlet-name>myServlet</servlet-name><servlet-class>org.wyy.mvc.originweb.MyServlet</servlet-class>
</servlet>
<!--配置servlet处理路径-->
<servlet-mapping><servlet-name>myServlet</servlet-name><url-pattern>/my/*</url-pattern>
</servlet-mapping>
1,/my/hello/query 请求
2,通过servlet-mapping找到对应的Servlet
3,在Servlet的service()中对url进行再判断和处理
public void service(ServletRequest req, ServletResponse res) throws ServletException, IOException{if(req.url.equals("/my/hello/add")){// 执行新增}if(req.url.equals("/my/hello/query")){// 执行业务操作: 查询Obj data = helloService.doQuery();// 其他后续处理,如:视图的处理}
}

上述代码可以看到,特定URL通过特定逻辑代码进行处理,这之间其实是有映射关系的。如果只配置一个Servlet,则Servlet中会掺杂很多业务,导致类臃肿混乱;如果在web.xml中一个实体配置一个Servlet,则配置繁杂。

思考:这里的service()方法体内部就好比现在的controller。SpringMVC其实是将通用、重复的代码进行封装。它是如何做的?

1.2 集成SpringMVC后

SpringMVC开发中,我们做哪些配置?

  1. 编写controller,配置requestMapping()
  2. 配置web.xml中的servelt,指定使用DispatcherServlet,并设置它能处理的URL

结合原生的请求处理流程和SpringMVC的开发模式,可以很轻松的看出:配置requestMapping就是配置请求映射,web.xml中的servelt只配置一个,说明这个Servlet是请求处理的入口,那么请求在这个Servlet中肯定做了委派转发,将不同请求转发给不同的controller类中。据此,看看源码:

1.2.1 DispatcherServlet源码

  1. 请求进入DispatcherServlet,DispatcherServlet没有重写service方法,所以调用父类HttpServlet.service

  1. FrameworkServlet作为DispatcherServlet的父类和HttpServlet的子类,它重写了service(httpreq,httpresp),所以调用FrameworkServlet.service()。这里if的判断不论分支,最终都会调用processRequest,因为super.service中根据请求方法的不同,将去调用doGet或doPost等,由于FrameworkServlet同样重写了父类的doGet等一系列方法,且在方法内统一调用processRequest,所以super.service最终还是调用processRequest。

  1. processRequest()使用模版模式,将请求的处理封装成doService(),交由子类重写

  1. DispatcherServlet作为FrameworkServlet的子类,重写了doService()方法,所以请求还是绕回了DispatcherServlet.doService(),这个方法中,doDispatch顾名思义,进行请求的分发。

  1. 前面提到过,多个controller,但web.xml中只配置了一个DispatcherServlet,那么这个类中一定做了请求转发的逻辑。

doDispatch方法中:

  1. getHandler()获取请求的处理器,也就是RequestMapping注解标注的方法。在controller中,我们使用RequestMapping注解标记方法,实际是完成了路径和方法之间的映射,Spring启动时扫描并获取映射关系,通过路径就很容易得到handler。
  2. 有了handler,如何使用handler处理请求呢,这里就需要adapter了。getHandlerAdapter()获取handler适配器。不同的handler需要不同的adapter进行处理
    1. RequestMapping注解标注的处理器,使用RequestMappingHandlerAdapter
    2. HttpRequestHandler类型的处理器,使用HttpRequestHandlerAdapter(见语雀:测试使用HttpRequestHandler实现Servlet)
  1. 适配器中,通过handle()方法,利用反射实现对handler处理器的调用,handler中完成对请求的处理。

看一下什么是handler适配器:

两个方法,1,判断是否支持适配传入的handler;2,处理请求。再看下接口实现类,以常用的RequestMapping注解的handler适配器为例,它的处理逻辑:

跟踪代码,最终看到:method.invoke(obj,args)

2 SpringMVC中请求的执行流程

  1. 调用servlet.service()方法 --> doService()
  2. 执行dispatcherServlet.doService() --> doDispatch()
    1. 根据请求URL,从handlerMapping中获取Handler,这个Handler是一个处理器执行链对象,这个对象中包含一个拦截器列表interceptorList

HandlerMapping中放的是 方法注解中的 URL --> 方法名

    1. 根据Handler获取 适配器 getHandlerAdapter(mappedHandler.getHandler());
    2. 遍历所有HandlerInterceptor的prehandle()
    3. 适配器对象调用具体的handle(),(通过反射调用 controller中的具体方法),返回ModelAndView视图对象
    4. 遍历所有HandlerInterceptor的PostHandler()
    5. 调用processDispatchResult(),处理派遣结果如渲染MV,方法中最后调用拦截器的afterCompletion()

3 总结

传统的servlet程序,请求到来时根据web.xml中的URL与servlet映射关系,直接找到相应的servlet进行处理。springMVC则可以看做是通过DispatcherServlet对请求做了二次转发,servlet中不再处理具体业务,而是将请求派遣给controller中的方法去执行。其中,方法定位的逻辑是通过请求的URL与controller方法上注解的路径进行比对,这些方法名和路径的映射关系被缓存到handlerMapping中。controller中的方法是HandlerAdapter通过反射执行的,执行前后分别执行过滤器相应的逻辑。业务方法执行完成,执行结果封装为一个MV模型视图对象,并提供相应的视图解析器对结果进行解析渲染。

欢迎评论,请勿转载!!!

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

数据结构-递归

用递归代替循环 假设工作中的你&#xff0c;需要写一个倒数程序。该程序接收一个数字&#xff0c;例如10&#xff0c;然后显示从10到0的数字。现在先暂停一下&#xff0c;选择一门编程语言来实现这个程序&#xff0c;做完以后&#xff0c;再往下阅读。或许你用了JavaScript&am…

数学建模--二分法

目录 二分法的基本原理 应用实例 求解方程根 查找有序数组中的元素 注意事项 Python代码示例 ​编辑 延伸 二分法在数学建模中的具体应用案例有哪些&#xff1f; 如何选择二分法的初始区间以确保收敛速度和精度&#xff1f; 在使用二分法求解方程时&#xff0c;如何…

排序算法2:直接选择排序与快速排序

目录 1.直接选择排序 1.1直接选择排序的优化 2.快速排序 2.1基准值的置位&#xff08;Hoare版&#xff09; 2.2挖坑法 2.3lomuto前后指针 前言 前面我们进入了排序算的讲解。今天我们将继续学习几种重要的排序思想&#xff0c;好&#xff0c;咱们三连上车开始今天的内容。…

ChatTTS文本转语音本地部署结合内网穿透实现远程使用生成AI音频

文章目录 前言1. 下载运行ChatTTS模型2. 安装Cpolar工具3. 实现公网访问4. 配置ChatTTS固定公网地址 前言 本篇文章主要介绍如何快速地在Windows系统电脑中本地部署ChatTTS开源文本转语音项目&#xff0c;并且我们还可以结合Cpolar内网穿透工具创建公网地址&#xff0c;随时随…

动态规划.

目录 &#xff08;一&#xff09;递归到动规的一般转化方法 &#xff08;二&#xff09;动规解题的一般思路 1. 将原问题分解为子问题 2. 确定状态 3. 确定一些初始状态&#xff08;边界状态&#xff09;的值 4. 确定状态转移方程 &#xff08;三&#xff09;能用动规解…

【网络】HTTP协议

目录 概述 URL 结构 urlencode&#xff08;URL编码&#xff09; urldecode&#xff08;URL解码&#xff09; 工具网址 HTTP请求 请求行 请求头 请求体 HTTP响应 状态行 响应头 响应体 个人主页&#xff1a;东洛的克莱斯韦克-CSDN博客 概述 HTTP协议是应用层协议…

TCP 三次握手建立连接

一开始&#xff0c;客户端和服务端都处于 CLOSE 状态。先是服务端主动监听某个端口&#xff0c;处于 LISTEN 状态 1. 第一次握手 客户端会随机初始化序号&#xff08;client_isn&#xff09;&#xff0c;将此序号置于 TCP 首部的「序号」字段中&#xff0c;同时把 SYN 标志位置…

略读ArrayList源码

ArrayList是Java集合框架中的一部分&#xff0c;底层是通过数组实现的&#xff0c;可以动态增长和缩减。 一、首先看成员变量 序列化ID定义。在Java中&#xff0c;如果一个类实现了Serializable接口&#xff0c;那么它的serialVersionUID就非常重要了。serialVersionUID用于确…

python 图片爬虫记录

感谢大家的点赞。再补充一点。 对于这个 url https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEjEqB5nighYsMZE7kexaVNJfxy3OkRutNEKatksw9u5f-ckHNROLzFyx2Uty3zYWNEaeOmzsljGr3eARiDWaM9DM8G2hPuPf8uZP0NO3kNUCnM2Cjb3ZKtLhJDBwqeR4ElpJ7ID5_wIHGQ/s200 这个url最…

Python进阶 JSON数据,pyecharts制图

目录 json数据格式的转换 什么是json json本质 注意 pyecharts快速入门 画一个最简单的折线图 使用全局配置选项优化折线图 总结 json数据格式的转换 什么是json 一种轻量级的数据交换格式&#xff0c;可以按json指定的格式去组织和封装数据 json本质 带有特定格式的…

汇川技术|Inoproshop基本使用方法:汇川指令库、库文件

哈喽&#xff0c;你好啊&#xff0c;我是雷工&#xff01; 本节熟悉了解汇川常用指令库的分类及概述&#xff0c;了解Inoproshop库文件&#xff1b; 以下为学习笔记。 01 指令简介与分类 可编程控制系统中&#xff0c;使CPU完成某种操作或实现某种功能的命令及多个命令的组合…

CCRC-DSA数据安全评估师:加快构建大网络安全工作格局

7月31日&#xff0c;第十二届ISC.AI互联网安全大会开幕式在北京国家会议中心隆重举行&#xff0c;本次大会以“构建大型安全防护模型&#xff0c;引领安全产业创新”为主题。 中央网络安全和信息化委员会办公室副主任、国家互联网信息办公室副主任王京涛出席并发表了重要讲话。…

语音平台调研

百度DuerOS开放平台 DuerOS是百度推出的对话式人工智能操作系统&#xff0c;即智能语音交互平台。DuerOS的技术架构包含“对话服务”和“技能框架”两大基础协议。两大协议连通起来的对话核心系统、智能设备开放平台和技能开放平台&#xff0c;构成了完整DuerOS的智能生态系统。…

C#初级——字典Dictionary

字典 字典是C#中的一种集合&#xff0c;它存储键值对&#xff0c;并且每个键与一个值相关联。 创建字典 Dictionary<键的类型, 值的类型> 字典名字 new Dictionary<键的类型, 值的类型>(); Dictionary<int, string> dicStudent new Dictionary<int, str…

Javascript常见算法(二)【学习】

动态规划 斐波那契数列&#xff1a; 经典的动态规划问题&#xff0c;每个数是前两个数的和。 斐波那契数列&#xff08;Fibonacci sequence&#xff09;是一个非常著名的数列&#xff0c;其中每个数是前两个数的和&#xff0c;序列以0和1开始。在JavaScript中&#xff0c;有多…

药厂子母钟系统,强抗干扰能力,满足复杂生产环境

在制药行业中&#xff0c;精确的时间同步对于确保药品生产的质量和合规性至关重要。药厂子母钟系统作为一种高度可靠的时间同步解决方案&#xff0c;不仅能够提供准确的时间信息&#xff0c;还具有强大的抗干扰能力&#xff0c;非常适合在复杂的生产环境中使用。本文将详细介绍…

[STM32]HAL库实现自己的BootLoader-BootLoader与OTA-STM32CUBEMX

目录 一、前言 二、BootLoader 三、BootLoader的实现 四、APP程序 五、效果展示 六、拓展 一、前言 听到BootLoader大家一定很熟悉&#xff0c;在很多常见的系统中都会存在BootLoader。本文将介绍BootLoader的含义和简易实现&#xff0c;建议大家学习前掌握些原理基础。 …

YOLOV8替换Lion优化器

YOLOV8替换Lion优化器 1 优化器介绍博客 参考bilibili讲解视频 论文地址&#xff1a;https://arxiv.org/abs/2302.06675 代码地址&#xff1a;https://github.com/google/automl/blob/master/lion/lion_pytorch.py """PyTorch implementation of the Lion …

C++初学(11)

不知不觉就第11篇了QWQ 11.1、指针和自由存储空间 之前提到了计算机程序在存储数据时必须跟踪的3个基本属性&#xff1a; &#xff08;1&#xff09;信息存储在何处&#xff1b; &#xff08;2&#xff09;存储的值为多少&#xff1b; &#xff08;3&#xff09;存储的信息…

未授权访问漏洞(非重点 中)

6.Hadoop 1.在 fofa 使用 port"8088" && app"Hadoop" 获取资源 2.打开后若无需登录,则存在漏洞 7.ActiveMQ 1.在 fofa 使用 body"ActiveMQ" && port"8161" 获取资源 2.打开后若点击登录,默认账户密码为 admin/adm…