资讯专栏INFORMATION COLUMN

[闹着玩-2]spring-mvc 主要流程

fuchenxuan / 2240人阅读

摘要:源码仓库本文仓库三层结构表现层模型业务层持久层工作流程用户前端控制器用户发送请求前端控制器后端控制器根据用户请求查询具体控制器后端控制器前端控制器处理后结果前端控制器视图视图渲染视图前端控制器返回视图前端控制器用户响应结

SpringMvc

【源码仓库】
【本文仓库】

三层结构

表现层

MVC模型

业务层

service

持久层

dao

工作流程

</>复制代码

  1. 用户->前端控制器:用户发送请求
  2. 前端控制器-> 后端控制器:根据用户请求查询具体控制器
  3. 后端控制器-->前端控制器:处理后结果
  4. 前端控制器--> 视图:视图渲染
  5. 视图-->前端控制器:返回视图
  6. 前端控制器--> 用户:响应结果

简单案例

依赖

</>复制代码

  1. 4.0.0
  2. com.huifer
  3. mySpringMvcBook
  4. 1.0-SNAPSHOT
  5. war
  6. UTF-8
  7. UTF-8
  8. 1.8
  9. 5.1.5.RELEASE
  10. 4.12
  11. org.springframework
  12. spring-beans
  13. ${spring.version}
  14. org.springframework
  15. spring-core
  16. ${spring.version}
  17. org.springframework
  18. spring-context
  19. ${spring.version}
  20. org.springframework
  21. spring-expression
  22. ${spring.version}
  23. org.springframework
  24. spring-aspects
  25. ${spring.version}
  26. aopalliance
  27. aopalliance
  28. 1.0
  29. org.springframework
  30. spring-webmvc
  31. ${spring.version}
  32. org.springframework
  33. spring-web
  34. ${spring.version}
  35. javax.servlet.jsp.jstl
  36. jstl
  37. 1.2
  38. javax.servlet
  39. servlet-api
  40. 2.5
  41. org.projectlombok
  42. lombok
  43. 1.18.4
  44. com.fasterxml.jackson.core
  45. jackson-databind
  46. 2.9.3
  47. com.fasterxml.jackson.core
  48. jackson-core
  49. 2.9.3
  50. com.fasterxml.jackson.core
  51. jackson-annotations
  52. 2.9.3
  53. src/main/java
  54. **/*.xml
  55. org.apache.maven.plugins
  56. maven-compiler-plugin
  57. 1.8
  58. 1.8
  59. UTF-8

spring-config

</>复制代码

web.xml

</>复制代码

  1. springmvc
  2. org.springframework.web.servlet.DispatcherServlet
  3. contextConfigLocation
  4. classpath:spring-mvc-config.xml
  5. 2
  6. springmvc
  7. /

controller

</>复制代码

  1. @Controller
  2. @RequestMapping("item")
  3. public class ItemController {
  4. @ResponseBody
  5. @GetMapping("/query")
  6. public ModelAndView query() throws Exception {
  7. List itemList = new ArrayList<>();
  8. itemList.add(new Item("吃的", 3.3, new Date()));
  9. itemList.add(new Item("玩的", 3.3, new Date()));
  10. ModelAndView modelAndView = new ModelAndView();
  11. modelAndView.addObject("itemList", itemList);
  12. modelAndView.setViewName("/WEB-INF/jsp/item.jsp");
  13. return modelAndView;
  14. }
  15. }

item.JSP

</>复制代码

  1. <%--
  2. Created by IntelliJ IDEA.
  3. User: huifer
  4. Date: 2019/3/10
  5. Time: 11:26
  6. To change this template use File | Settings | File Templates.
  7. --%>
  8. <%@ page language="java" contentType="text/html; charset=UTF-8"
  9. pageEncoding="UTF-8" isELIgnored="false" %>
  10. ItemList
  11. ITEM

  12. ${itemList}

根据tomcat的web.xml配置可以看到 /jsp的拦截是分开的 ,而我们直接用一个/*来拦截那么tomcat将不知道用什么来处理

spring-mvc大致流程源码翻阅

从配置文件中知道前端控制器DispatcherServlet

</>复制代码

  1. springmvc
  2. org.springframework.web.servlet.DispatcherServlet
  3. contextConfigLocation
  4. classpath:spring-mvc-config.xml
  5. 2

在org.springframework.web.servlet.DispatcherServlet 看下面两个方法

doService 将访问的数据接收到交给doDispatch

doDispatch 具体调度

</>复制代码

  1. protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception {
  2. HttpServletRequest processedRequest = request;
  3. HandlerExecutionChain mappedHandler = null;
  4. boolean multipartRequestParsed = false;
  5. WebAsyncManager asyncManager = WebAsyncUtils.getAsyncManager(request);
  6. try {
  7. try {
  8. ModelAndView mv = null;
  9. Object dispatchException = null;
  10. try {
  11. processedRequest = this.checkMultipart(request);
  12. multipartRequestParsed = processedRequest != request;
  13. // 加载handler
  14. mappedHandler = this.getHandler(processedRequest);
  15. if (mappedHandler == null) {
  16. this.noHandlerFound(processedRequest, response);
  17. return;
  18. }
  19. // handler适配器
  20. HandlerAdapter ha = this.getHandlerAdapter(mappedHandler.getHandler());
  21. String method = request.getMethod();
  22. boolean isGet = "GET".equals(method);
  23. if (isGet || "HEAD".equals(method)) {
  24. long lastModified = ha.getLastModified(request, mappedHandler.getHandler());
  25. if ((new ServletWebRequest(request, response)).checkNotModified(lastModified) && isGet) {
  26. return;
  27. }
  28. }
  29. if (!mappedHandler.applyPreHandle(processedRequest, response)) {
  30. return;
  31. }
  32. // 适配器执行操作
  33. mv = ha.handle(processedRequest, response, mappedHandler.getHandler());
  34. if (asyncManager.isConcurrentHandlingStarted()) {
  35. return;
  36. }
  37. this.applyDefaultViewName(processedRequest, mv);
  38. mappedHandler.applyPostHandle(processedRequest, response, mv);
  39. } catch (Exception var20) {
  40. dispatchException = var20;
  41. } catch (Throwable var21) {
  42. dispatchException = new NestedServletException("Handler dispatch failed", var21);
  43. }
  44. // 操作结果返回
  45. this.processDispatchResult(processedRequest, response, mappedHandler, mv, (Exception)dispatchException);
  46. } catch (Exception var22) {
  47. this.triggerAfterCompletion(processedRequest, response, mappedHandler, var22);
  48. } catch (Throwable var23) {
  49. this.triggerAfterCompletion(processedRequest, response, mappedHandler, new NestedServletException("Handler processing failed", var23));
  50. }
  51. } finally {
  52. if (asyncManager.isConcurrentHandlingStarted()) {
  53. if (mappedHandler != null) {
  54. mappedHandler.applyAfterConcurrentHandlingStarted(processedRequest, response);
  55. }
  56. } else if (multipartRequestParsed) {
  57. this.cleanupMultipart(processedRequest);
  58. }
  59. }
  60. }

handler怎么来

initHandlerMappings(context)

</>复制代码

  1. protected void initStrategies(ApplicationContext context) {
  2. this.initMultipartResolver(context);
  3. this.initLocaleResolver(context);
  4. this.initThemeResolver(context);
  5. this.initHandlerMappings(context);
  6. this.initHandlerAdapters(context);
  7. this.initHandlerExceptionResolvers(context);
  8. this.initRequestToViewNameTranslator(context);
  9. this.initViewResolvers(context);
  10. this.initFlashMapManager(context);
  11. }

</>复制代码

  1. private void initHandlerMappings(ApplicationContext context) {
  2. this.handlerMappings = null;
  3. if (this.detectAllHandlerMappings) {
  4. Map matchingBeans = BeanFactoryUtils.beansOfTypeIncludingAncestors(context, HandlerMapping.class, true, false);
  5. if (!matchingBeans.isEmpty()) {
  6. this.handlerMappings = new ArrayList(matchingBeans.values());
  7. AnnotationAwareOrderComparator.sort(this.handlerMappings);
  8. }
  9. } else {
  10. try {
  11. HandlerMapping hm = (HandlerMapping)context.getBean("handlerMapping", HandlerMapping.class);
  12. this.handlerMappings = Collections.singletonList(hm);
  13. } catch (NoSuchBeanDefinitionException var3) {
  14. }
  15. }
  16. if (this.handlerMappings == null) {
  17. // 读取默认配置文件
  18. this.handlerMappings = this.getDefaultStrategies(context, HandlerMapping.class);
  19. if (this.logger.isTraceEnabled()) {
  20. this.logger.trace("No HandlerMappings declared for servlet "" + this.getServletName() + "": using default strategies from DispatcherServlet.properties");
  21. }
  22. }
  23. }

默认配置文件

handler适配器也在配置文件中

</>复制代码

  1. protected HandlerAdapter getHandlerAdapter(Object handler) throws ServletException {
  2. if (this.handlerAdapters != null) {
  3. Iterator var2 = this.handlerAdapters.iterator();
  4. while(var2.hasNext()) {
  5. HandlerAdapter adapter = (HandlerAdapter)var2.next();
  6. // 是否能够适配
  7. if (adapter.supports(handler)) {
  8. return adapter;
  9. }
  10. }
  11. }
  12. throw new ServletException("No adapter for handler [" + handler + "]: The DispatcherServlet configuration needs to include a HandlerAdapter that supports this handler");
  13. }

HttpRequestHandlerAdapter

</>复制代码

  1. public class HttpRequestHandlerAdapter implements HandlerAdapter {
  2. public HttpRequestHandlerAdapter() {
  3. }
  4. // 判断是否是当前类支持的适配器
  5. public boolean supports(Object handler) {
  6. return handler instanceof HttpRequestHandler;
  7. }
  8. // 适配器执行操作
  9. @Nullable
  10. public ModelAndView handle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
  11. ((HttpRequestHandler)handler).handleRequest(request, response);
  12. return null;
  13. }
  14. public long getLastModified(HttpServletRequest request, Object handler) {
  15. return handler instanceof LastModified ? ((LastModified)handler).getLastModified(request) : -1L;
  16. }
  17. }

视图解析

</>复制代码

  1. private void processDispatchResult(HttpServletRequest request, HttpServletResponse response, @Nullable HandlerExecutionChain mappedHandler, @Nullable ModelAndView mv, @Nullable Exception exception) throws Exception {
  2. boolean errorView = false;
  3. if (exception != null) {
  4. if (exception instanceof ModelAndViewDefiningException) {
  5. this.logger.debug("ModelAndViewDefiningException encountered", exception);
  6. mv = ((ModelAndViewDefiningException)exception).getModelAndView();
  7. } else {
  8. Object handler = mappedHandler != null ? mappedHandler.getHandler() : null;
  9. mv = this.processHandlerException(request, response, handler, exception);
  10. errorView = mv != null;
  11. }
  12. }
  13. if (mv != null && !mv.wasCleared()) {
  14. // 这个地方在做渲染
  15. this.render(mv, request, response);
  16. if (errorView) {
  17. WebUtils.clearErrorRequestAttributes(request);
  18. }
  19. } else if (this.logger.isTraceEnabled()) {
  20. this.logger.trace("No view rendering, null ModelAndView returned.");
  21. }
  22. if (!WebAsyncUtils.getAsyncManager(request).isConcurrentHandlingStarted()) {
  23. if (mappedHandler != null) {
  24. mappedHandler.triggerAfterCompletion(request, response, (Exception)null);
  25. }
  26. }
  27. }

</>复制代码

  1. protected void render(ModelAndView mv, HttpServletRequest request, HttpServletResponse response) throws Exception {
  2. Locale locale = this.localeResolver != null ? this.localeResolver.resolveLocale(request) : request.getLocale();
  3. response.setLocale(locale);
  4. String viewName = mv.getViewName();
  5. View view;
  6. if (viewName != null) {
  7. // 视图解析器
  8. view = this.resolveViewName(viewName, mv.getModelInternal(), locale, request);
  9. if (view == null) {
  10. throw new ServletException("Could not resolve view with name "" + mv.getViewName() + "" in servlet with name "" + this.getServletName() + """);
  11. }
  12. } else {
  13. view = mv.getView();
  14. if (view == null) {
  15. throw new ServletException("ModelAndView [" + mv + "] neither contains a view name nor a View object in servlet with name "" + this.getServletName() + """);
  16. }
  17. }
  18. if (this.logger.isTraceEnabled()) {
  19. this.logger.trace("Rendering view [" + view + "] ");
  20. }
  21. try {
  22. if (mv.getStatus() != null) {
  23. response.setStatus(mv.getStatus().value());
  24. }
  25. // 视图的渲染函数渲染到页面上
  26. view.render(mv.getModelInternal(), request, response);
  27. } catch (Exception var8) {
  28. if (this.logger.isDebugEnabled()) {
  29. this.logger.debug("Error rendering view [" + view + "]", var8);
  30. }
  31. throw var8;
  32. }
  33. }

</>复制代码

  1. @Nullable
  2. protected View resolveViewName(String viewName, @Nullable Map model, Locale locale, HttpServletRequest request) throws Exception {
  3. if (this.viewResolvers != null) {
  4. Iterator var5 = this.viewResolvers.iterator();
  5. while(var5.hasNext()) {
  6. // 视图解析器
  7. ViewResolver viewResolver = (ViewResolver)var5.next();
  8. View view = viewResolver.resolveViewName(viewName, locale);
  9. if (view != null) {
  10. return view;
  11. }
  12. }
  13. }
  14. return null;
  15. }

spring-mvc请求具体流程图

</>复制代码

  1. 用户->前端控制器
  2. org.springframework.web.servlet.DispatcherServlet
  3. 用来接收响应以及返回响应结果:1.用户发送请求
  4. 前端控制器
  5. org.springframework.web.servlet.DispatcherServlet
  6. 用来接收响应以及返回响应结果 -> HandlerMapper : 2.根据url进行处理
  7. HandlerMapper -> 前端控制器
  8. org.springframework.web.servlet.DispatcherServlet
  9. 用来接收响应以及返回响应结果:3.handlerMapper处理结果给前端控制器
  10. 前端控制器
  11. org.springframework.web.servlet.DispatcherServlet
  12. 用来接收响应以及返回响应结果-> HandlerAdapter:4.给适配器确认具体的适配器,比如我们这里给了HttpRequestHandlerAdapter
  13. HandlerAdapter ->ModelAndView:5.用来执行业务操作
  14. ModelAndView ->HandlerAdapter: 6.执行完成给HandlerAdapter返回
  15. HandlerAdapter->前端控制器
  16. org.springframework.web.servlet.DispatcherServlet
  17. 用来接收响应以及返回响应结果:7.返回一个modelAndView
  18. 前端控制器
  19. org.springframework.web.servlet.DispatcherServlet
  20. 用来接收响应以及返回响应结果-> ViewResolver 视图解析器 :8.将第7步中的view进行解析
  21. ViewResolver 视图解析器->前端控制器
  22. org.springframework.web.servlet.DispatcherServlet
  23. 用来接收响应以及返回响应结果:9.解析结果返回
  24. 前端控制器
  25. org.springframework.web.servlet.DispatcherServlet
  26. 用来接收响应以及返回响应结果 -> View : 9.将model渲染到view中
  27. View->前端控制器
  28. org.springframework.web.servlet.DispatcherServlet
  29. 用来接收响应以及返回响应结果: 10.view 结果返回给前端控制器
  30. 前端控制器
  31. org.springframework.web.servlet.DispatcherServlet
  32. 用来接收响应以及返回响应结果 --> 用户:查看到完整的网页

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/73641.html

相关文章

  • spring-MVC源码解读(一)

    摘要:处理器是继前端控制器的后端控制器,在的控制下对具体的用户请求进行处理。由于涉及到具体的用户业务请求,所以一般情况需要程序员根据业务需求开发。 1、mcv整体架构和流程 showImg(https://segmentfault.com/img/bV55Qq?w=860&h=406); 用户发送请求至前端控制器 DispatcherServlet DispatcherServlet 收到...

    I_Am 评论0 收藏0
  • SSM框架运行机制

    摘要:框架搭建首先下载相应的包,对于包有两种方式使用创建依赖从而导入所需的包。总结主要进行页面的请求接受与响应。组件包括前端控制器,处理器映射器,处理器适配器,视图解析器,处理器,视图。 我之前的文章介绍了如何搭建SSH框架以及如何利用这一框架来进行web应用开发,最近我又接触了SSM框架即Spring+SpringMVC+Mybatis三大框架的整合,而且目前该框架就SSH框架而言使用的较...

    winterdawn 评论0 收藏0
  • SpringMVC入门笔记

    摘要:入门笔记简介是一种基于的实现了设计模式的请求驱动类型的轻量级框架,是系开源项目中的一个,和配合使用。配置在中需要添加使用的和映射规则。入门较快,而掌握起来相对较难。 SpringMVC入门笔记 1. 简介 Spring MVC是一种基于Java的实现了Web MVC设计模式的请求驱动类型的轻量级Web框架 ,是Spring系开源项目中的一个,和IoC配合使用。通过策略接口,Spring...

    zhaochunqi 评论0 收藏0
  • spring-mvc注解

    摘要:关键注解的关键注解主要有其中主要是用于标记该类是一个控制器,用于指示的哪一个类或方法来处理请求动作,即用于标识具体的处理器。默认已经装配了作为组件的实现类,而由使用,将请求信息转换为对象。 关键注解 springmvc的关键注解主要有@Controller/@RequestMapping/@RequestParam/@PathVariable/@RequestHeader/@Cooki...

    EdwardUp 评论0 收藏0

发表评论

0条评论

fuchenxuan

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<