原理
  
  本质是将DispatcherServlet及关联的Spring上下文环境的初始化工作织入Servlet的生命周期内,将外部WEB请求转换为Spring Bean能处理的形式,然后将处理后的结果借助于符合J2EE规范组件,呈现给客户端。
  
  步骤
  
  HttpServletBean: init(),内部调用initServletBean完成Servlet的初始化
  
  FrameworkServlet 重写 initServletBean()方法,方法内部调用 initWebApplicationContext()方法完成与DispatcherServlet关联的web应用上下文的初始化工作,应用环境的基础环境的初始化工作由子类DispatcherServlet的onRefresh方法完成。
  
  DispatcherServlet.onRefresh方法内部调用:initStrategies方法完成Springmvc需要的各种组件的初始化工作,至此springmvc可提供完整的外部应用访问功能。
  
  protected void initStrategies(ApplicationContext context) {
  
  initMultipartResolver(context);
  
  initLocaleResolver(context);
  
  initThemeResolver(context);
  
  initHandlerMappings(context);
  
  initHandlerAdapters(context);
  
  initHandlerExceptionResolvers(context);
  
  initRequestToViewNameTranslator(context);
  
  initViewResolvers(context);
  
  initFlashMapManager(context);
  
  }
  
  springmvc 默认配置文件:DispatcherServlet.properties
  
  Springmvc 客户端请求处理过程
  
  核心方法:doService,本质是对Serlvet接口的service方法的最终实现。
  
  doService:完成应用请求的数据封装封装,内部调用doDispatch方法完成请求调度调度处理
  
  源码分析:
  
  protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception {
  
  HttpServletRequest processedRequest = request;
  
  HandlerExecutionChain mappedHandler = null;
  
  boolean multipartRequestParsed = false;
  
  WebAsyncManager asyncManager = WebAsyncUtils.getAsyncManager(request);
  
  try {
  
  ModelAndView mv = null;
  
  Exception dispatchException = null;
  
  try {
  
  //判断外部请求是否是多部分请求(一般用于文件上传)
  
  processedRequest = checkMultipart(request);
  
  multipartRequestParsed = processedRequest != request;
  
  // 根据请求请求获取请求处理器:注意这里会返回对应的处理器以及潜在的拦截器
  
  mappedHandler = getHandler(processedRequest);
  
  if (mappedHandler == null || mappedHandler.getHandler() == null) {
  
  noHandlerFound(processedRequest, response);
  
  return;
  
  }
  
  // 获取能够支持请求处理器的处理器适配器
  
  HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());
  
  // Process last-modified header, if supported by the handler.
  
  String method = request.getMethod();
  
  boolean isGet = "GET".equals(method);
  
  if (isGet || "HEAD".equals(method)) {
  
  long lastModified www.yongshiyule178.com= ha.getLastModified(request, mappedHandler.getHandler());
  
  if (logger.isDebugEnabled()) {
  
  String requestUri = urlPathHelper.getRequestUri(request);
  
  logger.debug("Last-Modified www.xinghaiyule1.com value for [" + requestUri + "] is: " + lastModified);
  
  }
  
  if (new ServletWebRequest(request, response).checkNotModified(lastModified) && isGet) {
  
  return;
  
  }
  
  }
  
  //前置拦截器拦截请求
  
  if (!mappedHandler.applyPreHandle(processedRequest, response)) {
  
  return;
  
  }
  
  try {
  
  // 处理器适配器将请求传递给控制器处理,获取ModelView对象
  
  mv = ha.handle(processedRequest, response, mappedHandler.getHandler());
  
  }
  
  finally {
  
  if (asyncManager.isConcurrentHandlingStarted(dasheng178.com )) {
  
  return;
  
  }
  
  }
  
  //设置视图名
  
  applyDefaultViewName(request, mv);
  
  //后置拦截器拦截请求
  
  mappedHandler.applyPostHandle(processedRequest, response, mv);
  
  }
  
  catch (Exception ex) {
  
  dispatchException = ex;
  
  }
  
  //处理控制器返回的结果ModelAndView,主要是讲model的数据填充view,渲染后将数据返回给客户端显示
  
  processDispatchResult(processedRequest, response, mappedHandler, mv, dispatchException);
  
  }
  
  catch (Exception www.tiaotiaoylzc.com ex) {
  
  triggerAfterCompletion(processedRequest, response, mappedHandler, ex);
  
  }
  
  catch (Error err) {
  
  triggerAfterCompletionWithError(processedRequest, response, mappedHandler, err);
  
  }
  
  finally {
  
  if (asyncManager.isConcurrentHandlingStarted()) {
  
  // Instead of postHandle and afterCompletion
  
  mappedHandler.applyAfterConcurrentHandlingStarted(processedRequest, response);
  
  return;
  
  }
  
  // Clean up any resources used by a multipart request.
  
  if (multipartRequestParsed)www.mytxyl1.com {
  
  cleanupMultipart(processedRequest);
  
  }
  
  }
  
  }
  
  springmvc 请求响应:
  
  主要是将MappingHandler返回的ModelAndView种的Model数据填充到view,并借助于servlet容器(例如tomcat)呈现给客户端。
  
  AbstractView. render(Map<String, ?> model, HttpServletRequest request, HttpServletResponse response):
  
  //具体执行代码
  
  renderMergedOutputModel(mergedModel, request, response);
  
  案例分析:内部资源视图(例JSP) InternalResourceView. renderMergedOutputModel
  
  protected void renderMergedOutputModel(
  
  Map<String, Object>www.jtujbo.com model, HttpServletRequest request, HttpServletResponse response) throws Exception {
  
  //封装客户端可视的请求资源数据
  
  HttpServletRequest requestToExpose = getRequestToExpose(request);
  
  exposeModelAsRequestAttributes(model, requestToExpose);
  
  //客户端请求添加辅助信息
  
  exposeHelpers(requestToExpose);
  
  // 获取请求转发路径
  
  String dispatcherPath = prepareForRendering(requestToExpose, response);
  
  //获取目标资源的请求调度器
  
  RequestDispatcher rd = getRequestDispatcher(requestToExpose, dispatcherPath);
  
  if (rd == null) {
  
  throw new ServletException(www.yongshi123.cn"Could not get RequestDispatcher for [" + getUrl() +
  
  "]: Check that the corresponding file exists within your web application archive!");
  
  }
  
  //如果一个URL的内部请求则进行响应资源追加,例如网页内的图片链接等
  
  if (useInclude(requestToExpose, response)) {
  
  response.setContentType(getContentType());
  
  if (logger.isDebugEnabled()) {
  
  logger.debug("Including resource [" + getUrl() + "] in InternalResourceView '" + getBeanName() + "'");
  
  }
  
  rd.include(requestToExpose, response);
  
  }
  
  else {
  
  // Note: The forwarded resource is supposed to determine the content type itself.
  
  if (logger.isDebugEnabled(www.fengshen157.com)) {
  
  logger.debug("Forwarding to resource [" + getUrl(www.xtd912.com) + "] in InternalResourceView '" + getBeanName() + "'");
  
  }
  
  //repuest请求转发,说明springmvc采用的转发而不是重定向的方式
  
  rd.forward(requestToExpose, response);
  
  }

【spring】- springmvc 工作原理的更多相关文章

  1. springmvc工作原理和环境搭建

    SpringMVC工作原理     上面的是springMVC的工作原理图: 1.客户端发出一个http请求给web服务器,web服务器对http请求进行解析,如果匹配DispatcherServle ...

  2. [Java] SpringMVC工作原理之一:DispatcherServlet

    一.DispatcherServlet 处理流程 在整个 Spring MVC 框架中,DispatcherServlet 处于核心位置,它负责协调和组织不同组件完成请求处理并返回响应工作.在看 Di ...

  3. SpringMVC工作原理详解

    先来看一下什么是 MVC 模式 MVC 是一种设计模式. MVC 的原理图如下: SpringMVC 简单介绍 SpringMVC 框架是以请求为驱动,围绕 Servlet 设计,将请求发给控制器,然 ...

  4. SpringMVC 工作原理详解

    本文Github开源项目https://github.com/Snailclimb/JavaGuide,只供自己学习总结无商业用途,如有侵权,联系删除 先来看一下什么是 MVC 模式 MVC 是一种设 ...

  5. Spring MVC工作原理(好用版)

    Spring MVC工作原理 参考: SpringMVC工作原理 - 平凡希 - 博客园https://www.cnblogs.com/xiaoxi/p/6164383.html SpringMVC的 ...

  6. Spring Session工作原理

    本文首发于 vivo互联网技术 微信公众号 https://mp.weixin.qq.com/s/KCOFv0nRuymkX79-RZi9eg 作者:张正林 目录:1.引入背景2.使用方法3.工作流程 ...

  7. 笔面试复习(spring常用.jar包/事务/控制反转/bean对象管理和创建/springMVC工作原理/sql查询)

    ###spring常用jar包1.spring.jar是包含有完整发布模块的单个jar包.2.org.springframework.aop包含在应用中使用Spring的AOP特性时所需要的类.3.o ...

  8. springmvc工作原理以及源码分析(基于spring3.1.0)

    springmvc是一个基于spring的web框架.本篇文章对它的工作原理以及源码进行深入分析. 一.springmvc请求处理流程 二.springmvc的工作机制 三.springmvc核心源码 ...

  9. Spring MVC工作原理 及注解说明

    SpringMVC框架介绍 1) spring MVC属于SpringFrameWork的后续产品,已经融合在Spring Web Flow里面. Spring 框架提供了构建 Web 应用程序的全功 ...

  10. Spring MVC工作原理及源码解析(三) HandlerMapping和HandlerAdapter实现原理及源码解析

    1.HandlerMapping实现原理及源码解析 在前面讲解Spring MVC工作流程的时候我们说过,前端控制器收到请求后会调⽤处理器映射器(HandlerMapping),处理器映射器根据请求U ...

随机推荐

  1. 引用“kernel32”读写ini配置文件

    引用"kernel32"读写ini配置文件 unity ini kernel32 配置文件  引用"kernel32"读写ini配置文件 OverView ke ...

  2. 如何判断Map中的key或value类型

    在上班写工具类时,遇到了一个问题,将xml文件的节点都放入map容器中时,map的value也是一个map,导致取map的value时,需要判断这个value的数据类型,用到了一下说的这些知识: 对于 ...

  3. Netty源码分析第6章(解码器)---->第2节: 固定长度解码器

    Netty源码分析第六章: 解码器 第二节: 固定长度解码器 上一小节我们了解到, 解码器需要继承ByteToMessageDecoder, 并重写decode方法, 将解析出来的对象放入集合中集合, ...

  4. Linux重定向与管道

    程序执行时默认会打开3个流,标准输入.标准输出.标准错误. Redirection The shell interprets the symbols <,>, and >> a ...

  5. window搭建私有云,只要几分钟

    本文介绍如何在window搭建私有云网盘. 工具/原料:一台window系统电脑或者window服务器(vps),Xampp 安装包,可道云kodexplorer安装包 第一步,xampp安装 1.官 ...

  6. 基于Ubuntu+kodexplorer可道云的私有云网盘

    1.可用的服务器:组装PC机一台,操作系统为Ubuntu 14.04 LTS,无桌面环境,放在机房,使用远程终端进行访问.有安装了Apache2,运行着svn服务.内网IP地址为192.168.0.1 ...

  7. 开源ETL工具kettle系列之常见问题

    开源ETL工具kettle系列之常见问题 摘要:本文主要介绍使用kettle设计一些ETL任务时一些常见问题,这些问题大部分都不在官方FAQ上,你可以在kettle的论坛上找到一些问题的答案 1. J ...

  8. 重磅发布 | 黑镜调查:深渊背后的真相之「DDoS 威胁与黑灰产业调查报告」

    欢迎大家前往腾讯云+社区,获取更多腾讯海量技术实践干货哦~ 本文由云鼎实验室发表于云+社区专栏 本文经授权转载自 FreeBuf 2018年世界杯硝烟散尽,但关于她的话题却远远没有结束.说起世界杯,就 ...

  9. 机器学习基础 --- numpy的基本使用

    一.numpy的简介 numpy是Python的一种开源的数值计算扩展库.这种工具可用来存储和处理大型矩阵,比Python自身的嵌套列表(nested list structure)结构要高效的多(该 ...

  10. 工作小应用:EXCEL查找两列重复数据

    工作案例:excel存在A列.B列,需要找出B列没有A列的数据,具体做法如下(以office2007做案例): 1.点击 公式-定义名称 ,选中A列,填写名称“AAA”,选中B列,填写名称“BBB”: ...