百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

阿里Java二面:说说Spring MVC执行流程及原理?这样聊能吊打面试官

ccwgpt 2024-11-22 11:34 16 浏览 0 评论

面试找虐

博主之前每次去面试必问的问题:“讲一下spring mvc的执行流程以及常用组件的作用”;
记得第一次和面试官说了大概的流程是这样的:“服务器收到一个请求后会先去HandlerMapping中匹配url,找到url之后用HandlerAdapter 适配器去执行这个控制器(controller层),执行完之后返回一个modelAndView,然后通过视图解析器ViewResolver解析后得到一个view对象,这个view就是渲染好的jsp页面,返回给前端页面”;
(等了一会看着我)面试官:“   就这样。。。没啦?”;
懵逼的我:“嗯,没了”;
面试官:“那你回家等消息吧!”;

其实吧,博主说得没错,只是说少了,不够详细,如果是面试初级java开发,这样讲是没问题的,妥妥地给过了,但我是谁啊,我可是要成为架构师的人啊! 回答出这种答案,老板怎么放心把项目交给我呢。于是啊,吸取了教训,我就在家潜心研究,经过七七四十九天的研究、九九八十一次的练习,终于让我掌握了这门核心技术;

什么是Spring mvc

SpringMVC是一种web层mvc框架;是spring的一个模块,拥有spring的特性,通过策略接口,Spring框架是高度可配置的,而且包含多种视图技术,例如 JavaServer Pages(JSP)技术、Velocity、Tiles、iText 和 POI。Spring MVC 框架并不知道使用的视图,所以不会强迫您只使用 JSP 技术。Spring的MVC框架主要由DispatcherServlet、处理器映射、处理器(控制器)、视图解析器、视图组成。

都知道叫做spring mvc,mvc是什么东东呢?其实就是三层架构, 在过去架构都是两层的,就是浏览器通过servlet后直接访问JSP页面,也就是说所有的java代码都是在jsp页面里面写的,就连访问数据库也是在jsp页面,小型项目这样开发没什么问题,因为快嘛,但是大型项目是需要迭代的,是需要增加功能和模块的,你这么写的话,我后期维护起来很麻烦啊;而且java代码和html标签混搭,这样代码的可读性也不高;所以就衍生出了MVC三层架构

  • M : 表示mode,翻译成中文叫做模型,对应数据库;
  • V : 表示View,翻译为中文叫做视图,对应前端的页面,比如jsp
  • C : 表示 Controller,翻译是控制层,对应servlet的请求映射;

Spring MVC执行原理

终于到重点了,其实我在开头的时候已经答对了一半了,只是呢,少了一个叫做拦截器的东东,还有一些其他的细节,看下面这张图就明白了

Spring mvc 执行流程

  1. 服务器收到请求后,servlet 会将所有的请求都交给前端控制器(DispatcherServlet)处理;
  1. 前端控制器(DispatcherServlet)先调用HandlerMapping(处理器映射器),handlerMapping 根据请求url查找控制器(handler),返回一个 HandlerExecutionChain(处理器执行链) ,HandlerExecutionChain中除了控制器的信息之外,还有拦截器的信息;
  1. 接着将HandlerExecutionChain(处理器执行链)返回给 前端控制器(DispatcherServlet),调用适配器之前会先执行前置拦截器的代码,接着 DispatcherServlet 调用 HandlerAdapter(适配器)执行controller 层的代码内容,也就是带@Controller注解的类里面的代码,返回一个ModelAndView (逻辑视图);注意,这里的逻辑视图还不是真正的jsp页面;因为它还没经过渲染;再一次执行后置拦截器;
  2. 到这一步,DispatcherServlet就会调用ViewResolver(视图解析器)来解析ModelAndView 对象,得到一个View(视图),这个视图就是已经渲染好了的jsp页面了,然后把页面返回给前端之前再执行一次拦截器;

在mvc里面,最重要的无非就这几样 DispatcherServlet 、 HandlerMapping 、 HandlerExecutionChain 、 HandlerInterceptor拦截器、 HandlerAdapter、 ModelAndView、ViewResolver 、 View;别着急,我们一个个讲解它们的作用是什么;

DispatcherServlet

DispatcherServlet 的中文叫做前端控制器 ,也就叫中央调度器,用户发送的所有请求都会经过这个接口,由Servlet调用,是用来拦截所有请求的url的入口,并根据url地址分发到不同的controller ;在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_3_0.xsd" id="WebApp_ID" version="3.0">
  <display-name>SpringMVC_01</display-name>
  <!-- 注册中央调度器 -->
  <servlet>
    <servlet-name>springmvc</servlet-name>
    <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
    <!-- tomcat启动时自动创建servlet,数字越小优先级越高(>0) -->
    <load-on-startup>1</load-on-startup>
  </servlet>
  
  <servlet-mapping>
    <servlet-name>springmvc</servlet-name>
    <!-- 拦截所有请求 -->
    <url-pattern>/</url-pattern>
  </servlet-mapping>

</web-app>

HandlerMapping

HandlerMapping的作用就是解析请求链接,然后根据请求链接找到执行这个请求的类(HandlerMapping所说的handler,也就是我们写的Controller或是Action),也叫做控制器映射器,如果未匹配到控制器,则返回404错误;

HandlerExecutionChain

HandlerExecutionChain 是Handler的执行链,我叫它控制器执行链,包含控制器对象handler、拦截器 interceptor。所以HandlerExecutionChain 提供了getHandler、getInterceptors方法,配置文件中配置的interceptor都会加入到HandlerExecutionChain;需要注意的是,拦截器的列表里面除了我们配置的拦截器之外,还有一个springmvc自己维护的拦截器;

HandlerInterceptor

控制器的拦截器,一般都是作为扩展使用,一共有三个方法:

  • preHandle() :在业务处理器处理请求之前被调用。预处理,可以进行编码、安全控制、权限校验等处理;preHandle有一个boolen类型的返回值,返回 true 表示正常执行,返回 false 表示拦截请求.如果返回 false ,postHandle() 和 afterCompletion() 方法都不会执行;
  • postHandle() :在业务处理器处理请求执行完成后调用,生成视图之前执行。后处理(调用了Service并返回ModelAndView,但未进行页面渲染),有机会修改ModelAndView ;当controller层抛出异常后不会执行postHandle方法,
  • afterCompletion() :在DispatcherServlet完全处理完请求后被调用,可用于清理资源等。返回处理(已经渲染了页面);当controller层抛出异常后依然会执行afterCompletion方法

代码示例

package com.Spring.Boot.controller.Interceptor;
import org.springframework.stereotype.Component;
import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.ModelAndView;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
 
/**
 * springmvc拦截器
 *
 */
@Component
public class MvcInterceptor implements HandlerInterceptor {
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        System.out.println("处理器处理请求之前被调用");
        // 返回true正常执行,返回false 表示拦截请求,
        return true;
    }
 
    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
        // preHandle方法返回false则不会执行到这里
        System.out.println("处理器处理请求执行完成后调用");
    }
 
    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        // preHandle方法返回false则不会执行到这里
        System.out.println("DispatcherServlet完全处理完请求后被调用");
 
    }

}

打印结果

HandlerAdapter

HandlerAdapter字面上的意思就是处理适配器,它的作用用一句话概括就是调用具体的方法对用户发来的请求来进行处理。当handlerMapping获取到执行请求的controller时,DispatcherServlte会根据controller对应的controller类型来调用相应的HandlerAdapter来进行处理。

执行过程 : DispatcherServlte会根据handlerMapping传过来的controller与已经注册好了的HandlerAdapter一一匹配,看哪一种HandlerAdapter是支持该controller类型的,如果找到了其中一种HandlerAdapter是支持传过来的controller类型,那么该HandlerAdapter会调用自己的handle方法

ModelAndView

从名字上看ModelAndView中的Model代表模型,View代表视图,这里的模型就是在controller层中ModelMap中添加的值;

ModelAndView对象有两个作用:

作用一

设置转向地址,如下所示(这也是ModelAndView和ModelMap的主要区别)

ModelAndView mv = new ModelAndView();
mv.setViewName("statisticalform/popStatistics/onloadPopStatisticsList");

作用二

用于传递控制方法处理结果数据到结果页面,也就是说我们把需要在结果页面上需要的数据放到ModelAndView对象中即可,

他的作用类似于 request对象的setAttribute方法的作用,用来在一个请求过程中传递处理的数据。

通过以下方法向页面传递参数:

addObject(String key,Object value);   
mv.addObject("list", list);

在页面上可以通过el变量方式$key或者bboss的一系列数据展示标签获取并展示ModelAndView中的数据。

ViewResolver

视图解析器,ViewResolver的主要作用是把一个逻辑上的视图 ModelAndView 解析为一个真正的view视图;

View

视图的作用是渲染模型数据,将模型里的数据以某种形式呈现给客户端。视图对象可以是常见的JSP,还可以是Excel或PDF等形式不一的媒体。为了实现视图模型和具体实现技术的解耦;

转载于:https://blog.csdn.net/qq_27184497/article/details/117233063

相关推荐

VUE3前端开发入门系列教程二:使用iView框架辅助开发

1、安装iView新框架,支持VUE3npminstallview-ui-plus2、编辑src/main.js,添加以下内容,导入js和css到项目importViewUIPlusfrom...

万能前端框架uni app初探03:底部导航开发

前言本节我们使用uniapp的底部导航功能,点击不同tab会显示不同页面,这个功能在实际项目开发中几乎是必备的。一、基础知识1.tabBar如果应用是一个多tab应用,可以通过tabBar配...

Rust Web 开发框架,前端你可以选择哪个?

Rust构建一切。在如今流行的语言中,Rust可谓是将构建和高效作为自己优美的身姿在大众视野中脱颖而出。它是一门赋予每个人构建可靠且高效软件能力的语言。它有什么特性呢?高性能。Rust速度惊人且内...

连载:前端开发中纠结的Javascript框架(上)

如今,前端开发有着许许多多的框架和库。其中一些好用,一些却不尽人意。通常我们会习惯性运用某一概念,模块或句法。事实上,并没有什么万能工具。这篇文章是关于未来框架的发展趋势——那就是没有框架!我从以下几...

前端开发框架的演进架构:提升用户体验和开发效率

前端开发框架是现代Web应用开发的重要工具,它不仅可以帮助开发者构建复杂的用户界面,还能够提升用户体验和开发效率。随着Web技术的不断发展,前端开发框架也在不断演进,为开发者提供了更丰富、更高效的工具...

Google应用Mesh-TensorFlow框架,让CNN也能处理超高分辨率图像

为了要处理超高分辨率医疗图像数据,Google开发了一种空间数据分区(SpatialPartition)技术,在不牺牲图像分辨率的条件下,分析超高分辨率图像。Google使用Mesh-TensorF...

大模型安全挑战加剧:框架层漏洞成新靶心

近日,360数字安全集团发布了一份关于大模型安全漏洞的报告,揭示了当前大模型及围绕其构建的框架和应用中存在的严重安全问题。报告显示,360近期研究发现了近40个大模型相关的安全漏洞,其中既包括二进制内...

Keras 3.0正式发布:可用于TensorFlow、JAX和PyTorch

机器之心报道编辑:陈萍经过5个月的更新迭代,Keras3.0终于来了。「大新闻:我们刚刚发布了Keras3.0版本!」Keras之父FrancoisChollet在X上激动的...

TensorFlow和Keras入门必读教程(tensorflow与keras版本对应)

导读:本文对TensorFlow的框架和基本示例进行简要介绍。作者:本杰明·普朗什(BenjaminPlanche)艾略特·安德烈斯(EliotAndres)来源:华章科技01TensorFlo...

谷歌官方回应“TensorFlow遭弃”:还在投资开发,将与JAX并肩作战

鱼羊发自凹非寺量子位|公众号QbitAI终于,谷歌出面回应“TensorFlow遭弃”传闻:我们将继续致力于将TensorFlow打造为一流机器学习平台,与JAX并肩推动机器学习研究。这段时...

2025 年的PHP :现代 Web 开发的强大引擎

程序员还在吐槽PHP过时?2025年的PHP8.4直接封神了。看看最近更新的属性钩子、强类型系统,加上Laravel这些框架,老语言早就脱胎换骨。十年前说PHP弱类型容易崩代码的,现在脸疼不?联合类...

前端内卷终结者?htmx如何让开发者告别200行JS只做一个按钮

当你用React写一个点赞按钮需要引入3个状态管理库、编写80行JSX和120行钩子函数时,htmx只需要一行HTML:<buttonhx-post="/like"hx-sw...

NativePHP桌面版V1.0正式发布(元气桌面电脑版下载)

导读:各位小伙伴,使用PHP构建桌面级系统的利器,NativePHP来了。概述NativePHP是一个用于使用PHP构建桌面应用的框架。它允许PHP开发人员使用熟悉的工具和技术创建跨平台的原生应用...

PHP Laravel框架底层机制(php基本框架)

当然可以,Laravel是最受欢迎的PHP框架之一,以优雅的语法和丰富的生态而闻名。尽管开发体验非常“高端”,它的底层其实是由一系列结构清晰、职责分明的组件构成的。下面我从整体架构、核心流程、...

PHP框架之Laravel框架教程:2. 控制器、路由、视图简单介绍

2.控制器、路由、视图简单介绍我们先建立控制器,目录是:app/Http/Controllers,新建控制器Ding.php,代码如下:Ding.php:<?phpnamespaceA...

取消回复欢迎 发表评论: