2.Spring MVC工作流程

需求:前端浏览器请求 url http://localhost:8080/demo/handle01 ,前端⻚⾯显示后台服务器的时间
开发过程
 
1 )配置 DispatcherServlet 前端控制器
 
 
2 )开发处理具体业务逻辑的 Handler @Controller @RequestMapping
 
3 xml 配置⽂件配置 controller 扫描,配置 springmvc 三⼤件
 
 
4 )将 xml ⽂件路径告诉 springmvc DispatcherServlet
 
 
创建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 请求处理流程
 

 

流程说明

第⼀步:⽤户发送请求⾄前端控制器 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

解决方案2

输出机制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";
    }

 

评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值