zoukankan      html  css  js  c++  java
  • (spring-第21回【MVC基础篇】)SpringMVC一点就通

    概述

    • Spring MVC通过一套MVC注解,让POJO变成处理请求的控制器,无需实现任何接口,同时,SpringMVC还支持REST风格的URL请求:注解驱动和REST风格的Spring MVC是Spring 3.0最出彩的功能之一。
    • Spring MVC框架围绕 DispatcherServlet这个核心展开,DispatcherServlet是SpringMVC的总导演,它负责截获请求,并将其分派给相应的处理器处理。
    • 和大多数WEB MVC一样,Spring MVC通过一个前端Servlet接收所有的请求,并将具体工作委托给其他组件进行处理,DispatcherServlet就是Spring MVC的前端Servlet。

    本文主要讲了一些什么

    Spring MVC的基本原理  -->

    用一个简单的示例对基本原理有个具体了解 -->

    分别描述MVC的几个重量级角色:过滤请求、绑定请求、封装对象、处理XML和JSON、处理模型数据。

    本篇文章重在知识点整合,作为读书笔记,语言简洁,删繁就简,由浅入深,引人入胜。通读本文,你将对Spring MVC有一个深入的了解。如果读,请深读。如果爱,请读完。

    基本原理

    基本原理及入门示例已在微信订阅号上发布,为防止原创性的冲突检测,故删去这一部分,详情扫码关注查看:

    过滤请求

    使用@RequestMapping可以过滤请求URL、请求参数、请求方法,请求头这4个方面的信息项。比如:

    绑定请求信息到方法入参中

    Spring MVC会将HTTP请求的信息绑定到相应的方法入参中,并根据方法返回值类型作出相应的后续处理。比如:

    将请求信息封装成对象

    使用HttpMessageConverter<T>可以将请求信息转换成一个对象,或者将对象输出为响应信息。(T就是对象类型)。

    两把利剑:

    1.使用RequestBody、ResponseBody。

    2.使用HttpEntity<T>/ResponseEntity<T>

    示例1:

    客户端:

    服务端:

    示例2:

    客户端:

    服务端:

    与RequestBody、ResponseBody不同的是,HttpEntity不仅可以访问请求、响应报文体的数据,还可以访问请求和响应报文头的数据。

    示例1:

    客户端:

    服务端:

    示例2:

    客户端:

    服务端:

    怎么做到的?

    HttpMessageConverter<T>提供了众多的实现类,它组成了一个功能强大、用途广泛的HttpMessageConverter<T>家族。

    比如:

      StringHttpMessageConverter负责将请求信息转换成字符串;

      FormHttpMessageConverter负责将表单数据读取到MultiValueMap中;

      还有诸如读写XML、Resource、JSON等等对应的实现类。

      AnnotationMethodHandlerAdapter适配器默认装配了基本的Converter(String、ByteArray、Source、XmlAwareForm),容器会根据需要自动调用不同的HttpMessageConverter实现类。

      如果不能满足需求,那么继续在xml中配置其他的converter:

    处理XML和JSON

    从上文可知,HttpMessageConverter有一个很强大的家族,能够处理各种格式的请求及响应。当然也可以处理XML和JSON。

    处理XML和JSON的Converter分别是:

    MarsHallingHttpMessageConverter(处理xml)。

     

    Jaxb2RootElementHttpMessageConverter(同上,底层使用JAXB)。

     

    MappingJacksonHttpMessageConverter(处理JSON)。

    按照前文的思路,首先要在配置文件中装配好相应的Converter,但是还有其他必要的步骤:

    步骤1:XML中配置Converter以及Marshaller:

    步骤2:测试客户端发出请求:

    步骤3:响应(服务)端:

    步骤4:User的注解:

    步骤5:用tcptrace监听测试

    其实,如果把第92行和第93行代码的头部格式换成json,就变成传送json格式的数据了。这个时候,用tcptrace监听传送json的结果:

    可以看到,请求中,我们的报文头格式变成了json,报文体也是json格式的。而响应中报文体也是json格式的。

    为什么我直接截图了json格式的测试结果而没有测试请求xml的测试结果呢?

    因为在xml测试时,报出了415 Unsupported media type的错误,尚未发现原因。

    但是思路就是这样了。 读者如果知道“不支持此类型”的原因,欢迎评论留言。

     

     

    处理模型数据

     

     

    我们在Controller控制器中接收请求参数,把处理后得到的数据封装起来,然后返回到一个JSP,在JSP中就可以获取封装的数据。那么,在这个例子中,封装的数据就是模型数据,而JSP就是渲染模型数据的视图。这在MVC框架中是最重要的步骤。

    在控制器方法中输出模型数据有四种方法:

    方法一:ModelAndView

    在控制器方法中新建ModelAndView对象,把“视图”和“模型数据”封装到该对象中,然后返回该对象即可。

    方法二:@ModelAttribute

    加到入参的前面,就自动把入参绑定为模型数据;加到方法A的前面,那么Spring MVC就会在执行任何目标处理方法之前,先执行方法A(BC等等),然后把这些加了标注的方法的返回值绑定为模型数据。

    举例1:

    Spring MVC会先把模型数据储存到ServletRequest的属性列表中(使用setAttribute保存)。

    然后在返回的jsp(视图对象)中,使用${user.userName}来访问user的属性。

    举例2:

    在执行handle62之前,spring扫描到有方法头部标注了@ModelAttribute,那么,spring就会先去执行getUser,并把返回值user作为模型数据,键为”user”。

    这时执行handle62,入参前面的@ModelAttribute键为”user”,正好匹配了刚封装的模型数据的键,那么就把模型数据赋值给入参User user,然后285行对user做了覆盖操作,最后交给视图层处理。

    方法三:Map and Model

    如果处理方法的入参类型为Map或者Model(或者具备二者功能的对象如:ModelMap),那么spring会将隐含的模型数据传给该入参,在处理方法中,我们可以从该入参中取出隐含的模型数据,也可以继续添加模型数据。

    标注了@ModelAttribute的方法的返回值(getUser方法返回的user)就是隐含的模型数据。

    由于handle63的入参为ModelMap类型,所以,spring会把user这个隐含模型数据授予modelMap,在300行,我们通过该入参取出模型数据,在302行,我们通过该入参继续添加模型数据。

    方法四:@SessionAttributes

    如果Controller类的头部标注了@SessionAttributes(”xxx”),那么spring会把键为xxx的模型数据暂存到HttpSession中,以便多个请求之间可以共享该数据。

    ... ...

    61行的标注表示:后面处理器在处理任何方法时,看到属性名为user的模型属性就会把它存到session中共享。由于328行把user作为了模型数据,那么这个user就放到了session中。

    当转发请求到handle72时,就可以从modelMap中拿到该对象。

    当使用完成时,在338行表示把该对象从session中移除。否则它会一直存在。

    然而程序是跑不起来的,会报HttpSessionRequiredException。为什么呢?

    这是因为,如果有了61行标注的会话属性user,那么spring就会尝试从会话中获取该属性,然后把它赋给入参。

    也就是说,spring看到61行的标注,会从会话中找user为键的模型数据,找到后把它赋值给334行的modelMap。

    但是现在从代码看来,在执行61行标注之前,会话中并没有键为user的模型数据。

    所以,解决方法很简单:

    只需要像方法二那样,加一个:

     

    就好了。这样,在执行61行之前,先执行getUser,把user作为隐含模型数据。那么执行61行时,该会话中就已经有user了。

    当你读到这里,你已经百毒不侵了。亲,不点个赞吗?

  • 相关阅读:
    Deployment descriptor
    实体、list 、xml之间的转化
    关于C# 汉字转拼音问题
    NPoco学习笔记(1)
    SQL(二)
    SQL(一)
    sobel算子及cvSobel
    图像的平滑处理
    erase的用法
    int main(int argc, char* argv[ ])
  • 原文地址:https://www.cnblogs.com/mesopotamia/p/5534060.html
Copyright © 2011-2022 走看看