需求:前端浏览器请求 url : http://localhost:8080/demo/handle01 ,前端⻚⾯显示后台服务器的时间


开发过程


 


1 )配置 DispatcherServlet 前端控制器


 



springmvc Service层业务逻辑代码 springmvc工作流程_MVC


 


2 )开发处理具体业务逻辑的 Handler ( @Controller 、 @RequestMapping )


 



springmvc Service层业务逻辑代码 springmvc工作流程_spring_02


3 ) xml 配置⽂件配置 controller 扫描,配置 springmvc 三⼤件


 



springmvc Service层业务逻辑代码 springmvc工作流程_xml_03


 


4 )将 xml ⽂件路径告诉 springmvc ( DispatcherServlet )


 


 



springmvc Service层业务逻辑代码 springmvc工作流程_MVC_04


创建web-app项目


在pom.xml中引入spring-webmvc依赖, tomcat插件,


<dependency>  <groupId>org.springframework</groupId>  <artifactId>spring-webmvc</artifactId>  <version>5.3.5</version></dependency>

<build>
  <plugins>
    <plugin>
      <groupId>org.apache.tomcat.maven</groupId>
      <artifactId>tomcat7-maven-plugin</artifactId>
      <version>2.2</version>
      <configuration>
        <port>8080</port>
        <path>/</path>
      </configuration>
    </plugin>
  </plugins>
</build>

 

2.1 Spring MVC 请求处理流程


 



springmvc Service层业务逻辑代码 springmvc工作流程_xml_05


 

流程说明


第⼀步:⽤户发送请求⾄前端控制器 DispatcherServlet

第⼆步: DispatcherServlet 收到请求调⽤ HandlerMapping 处理器映射器


第三步:处理器映射器根据请求 Url 找到具体的 Handler (后端控制器),⽣成处理器对象及处理器拦截


 


器 ( 如果 有则⽣成 ) ⼀并返回 DispatcherServlet

第四步: DispatcherServlet 调⽤ HandlerAdapter 处理器适配器去调⽤ Handler


 


第五步:处理器适配器执⾏ Handler


 


第六步: Handler 执⾏完成给处理器适配器返回 ModelAndView


 


第七步:处理器适配器向前端控制器返回 ModelAndView , ModelAndView 是 SpringMVC 框架的⼀个


底层对 象,包括 Model 和 View


 


第⼋步:前端控制器请求视图解析器去进⾏视图解析,根据逻辑视图名来解析真正的视图。


 


第九步:视图解析器向前端控制器返回 View


 


第⼗步:前端控制器进⾏视图渲染,就是将模型数据(在 ModelAndView 对象中)填充到 request 域


 


第⼗⼀步:前端控制器向⽤户响应结果


 

2.2 Spring MVC 九⼤组件


 


HandlerMapping (处理器映射器)


HandlerMapping 是⽤来查找 Handler 的,也就是处理器,具体的表现形式可以是类,也可以是


⽅法。⽐如,标注了 @RequestMapping 的每个⽅法都可以看成是⼀个 Handler 。 Handler 负责具


体实际的请求处理,在请求到达后, HandlerMapping 的作⽤便是找到请求相应的处理器


 


Handler 和 Interceptor.


HandlerAdapter (处理器适配器) HandlerAdapter 是⼀个适配器。因为 Spring MVC 中 Handler 可以是任意形式的,只要能处理请


求即可。但是把请求交给 Servlet 的时候,由于 Servlet 的⽅法结构都是


doService(HttpServletRequest req,HttpServletResponse resp) 形式的,要让固定的 Servlet 处理


⽅法调⽤ Handler 来进⾏处理,便是 HandlerAdapter 的职责。


 


HandlerExceptionResolver


HandlerExceptionResolver ⽤于处理 Handler 产⽣的异常情况。它的作⽤是根据异常设置


ModelAndView ,之后交给渲染⽅法进⾏渲染,渲染⽅法会将 ModelAndView 渲染成⻚⾯。


 


ViewResolver


ViewResolver 即视图解析器,⽤于将 String 类型的视图名和 Locale 解析为 View 类型的视图,只有⼀


个 resolveViewName() ⽅法。从⽅法的定义可以看出, Controller 层返回的 String 类型视图名


viewName 最终会在这⾥被解析成为 View 。 View 是⽤来渲染⻚⾯的,也就是说,它会将程序返回


的参数和数据填⼊模板中,⽣成 html ⽂件。 ViewResolver 在这个过程主要完成两件事情:


ViewResolver 找到渲染所⽤的模板(第⼀件⼤事)和所⽤的技术(第⼆件⼤事,其实也就是找到


视图的类型,如 JSP )并填⼊参数。默认情况下, Spring MVC 会⾃动为我们配置⼀个


InternalResourceViewResolver, 是针对 JSP 类型视图的。


 


RequestToViewNameTranslator


RequestToViewNameTranslator 组件的作⽤是从请求中获取 ViewName. 因为 ViewResolver 根据


ViewName 查找 View ,但有的 Handler 处理完成之后 , 没有设置 View ,也没有设置 ViewName ,


便要通过这个组件从请求中查找 ViewName 。


 


LocaleResolver


ViewResolver 组件的 resolveViewName ⽅法需要两个参数,⼀个是视图名,⼀个是 Locale 。


LocaleResolver ⽤于从请求中解析出 Locale ,⽐如中国 Locale 是 zh-CN ,⽤来表示⼀个区域。这


个组件也是 i18n 的基础。


 


ThemeResolver


ThemeResolver 组件是⽤来解析主题的。主题是样式、图⽚及它们所形成的显示效果的集合。


Spring MVC 中⼀套主题对应⼀个 properties ⽂件,⾥⾯存放着与当前主题相关的所有资源,如图


⽚、 CSS 样式等。创建主题⾮常简单,只需准备好资源,然后新建⼀个 “ 主题名 .properties” 并将资


源设置进去,放在 classpath 下,之后便可以在⻚⾯中使⽤了。 SpringMVC 中与主题相关的类有


ThemeResolver 、 ThemeSource 和 Theme 。 ThemeResolver 负责从请求中解析出主题名,


ThemeSource 根据主题名找到具体的主题,其抽象也就是 Theme ,可以通过 Theme 来获取主题和


具体的资源。


 


MultipartResolver


MultipartResolver ⽤于上传请求,通过将普通的请求包装成 MultipartHttpServletRequest 来实


现。 MultipartHttpServletRequest 可以通过 getFile() ⽅法 直接获得⽂件。如果上传多个⽂件,还


可以调⽤ getFileMap() ⽅法得到 Map<FileName , File> 这样的结构, MultipartResolver 的作⽤就


是封装普通的请求,使其拥有⽂件上传的功能。


 


FlashMapManager


FlashMap ⽤于重定向时的参数传递,⽐如在处理⽤户订单时候,为了避免重复提交,可以处理完


post 请求之后重定向到⼀个 get 请求,这个 get 请求可以⽤来显示订单详情之类的信息。这样做虽然


可以规避⽤户重新提交订单的问题,但是在这个⻚⾯上要显示订单的信息,这些数据从哪⾥来获得


呢?因为重定向时么有传递参数这⼀功能的,如果不想把参数写进 URL (不推荐),那么就可以通


过 FlashMap 来传递。只需要在重定向之前将要传递的数据写⼊请求(可以通过 ServletRequestAttributes.getRequest() ⽅法获得)的属性 OUTPUT_FLASH_MAP_ATTRIBUTE


中,这样在重定向之后的 Handler 中 Spring 就会⾃动将其设置到 Model 中,在显示订单信息的⻚⾯

上就可以直接从 Model 中获取数据。 FlashMapManager 就是⽤来管理 FalshMap


 


如何解决/拦截静态资源这件事?


 


<!--  方式一:带后缀 *.action,*.do,*.aa,比较精确,方便,在以前和现在在企业中都有很大的使用比例      方式二:/  rest风格,不会拦截.jsp,但是会拦截.html等静态资源文件      (静态资源文件:除了servlet和jsp之外的js,css,.png等)      为什么配置为/会拦截静态资源,      因为tomcat中有一个web.xml(父),你的项目中有一个web.xml(子),是一个继承关系      父web.xml中有一个DefaultServlet,url-pattern 是一个/
      此时我们自己的web.xml中也配置了/,复写了父web.xml的配置
       为什么不拦截.jsp呢?
               因为父web.xml中有一个JspServlet,这个servlet拦截.jsp文件,而我们并没有覆写这个配置,
               所以springmvc此时不拦截jsp,jsp的处理交给了tomcat
        如何解决/拦截静态资源这件事?

      方式三:/* 拦截所有,包括jsp
-->
   <!-- 匹配拦截规则的url请求,进入springmvc框架处理-->
  <url-pattern>/</url-pattern>

 解决方案1

springmvc Service层业务逻辑代码 springmvc工作流程_spring_06

解决方案2

springmvc Service层业务逻辑代码 springmvc工作流程_MVC_07

输出机制Map、Model和ModelMap

/**
      * SpringMVC在handler方法上传入Map、Model和ModelMap参数,并向这些参数中保存数据(放入到请求域),都可以在页面获取到
      *
      * 它们之间是什么关系?
      * 运行时的具体类型都是BindingAwareModelMap,相当于给BindingAwareModelMap中保存的数据都会放在请求域中
      *
      *  Map(jdk中的接口)        Model(spring的接口)
      *
      *  ModelMap(class,实现Map接口)
      *
      *
      *
      *
      *              BindingAwareModelMap继承了ExtendedModelMap,ExtendedModelMap继承了ModelMap,实现了Model接口
      *            所以Map、Model和ModelMap底层都是BindingAwareModelMap
      */ 
     /**
      * 直接声明形参ModelMap,封装数据
      * url: http://localhost:8080/demo/handle11
      *
      * =================modelmap:class org.springframework.validation.support.BindingAwareModelMap
      */
     @RequestMapping("/handle11")
     public String handle11(ModelMap modelMap) {
         Date date = new Date();// 服务器时间
         modelMap.addAttribute("date",date);
         System.out.println("=================modelmap:" + modelMap.getClass());
         return "success";
     }     /**
      * 直接声明形参Model,封装数据
      * url: http://localhost:8080/demo/handle12
      * =================model:class org.springframework.validation.support.BindingAwareModelMap
      */
     @RequestMapping("/handle12")
     public String handle12(Model model) {
         Date date = new Date();
         model.addAttribute("date",date);
         System.out.println("=================model:" + model.getClass());
         return "success";
     }     /**
      * 直接声明形参Map集合,封装数据
      * url: http://localhost:8080/demo/handle13
      * =================map:class org.springframework.validation.support.BindingAwareModelMap
      */
     @RequestMapping("/handle13")
     public String handle13(Map<String,Object> map) {
         Date date = new Date();
         map.put("date",date);
         System.out.println("=================map:" + map.getClass());
         return "success";
     }