Spring mvc

Spring mvc 框架

  • DispatcherServlet

    前端控制器 ---- 整个流程控制的中心,由它调用其它组件处理用户的请求

  • 三大组件

    • HandlerMapping

      处理器映射器 ---- 根据用户请求url找到处理器(Handler),springmvc提供了不同的映射器实现不同的映射方式,例如:配置文件方式,实现接口方式,注解方式等

    • HandlAdapter

      处理器适配器 ---- 通过HandlerAdapter对处理器进行执行,通过扩展适配器可以对更多类型的处理器进行执行

    • ViewResolver

      视图解析器 ---- 负责将处理结果生成View视图,View Resolver首先根据逻辑视图名解析成物理视图名即具体的页面地址,再生成View视图对象,最后对View进行渲染将处理结果通过页面展示给用户

  • Handler

    处理器 ---- 继DispatcherServlet前端控制器的后端控制器,在DispatcherServlet的控制下Handler对具体的用户请求进行处理

    默认支持的参数类型

    • HttpServletRequest

      通过request对象获取请求信息

    • HttpServletResponse

      通过response处理响应信息

    • HttpSession

      通过session对象得到session中存放的对象

  • View

    视图 ---- springmvc框架提供了很多的View视图类型的支持,包括:jstlView、freemarkerView、pdfView等。我们最常用的视图就是jsp

    视图对象

    • ModelAndView

    • Model

      Model是一个接口,在参数里直接声明model即可,Model对象可以向页面传递数据,View对象则可以使用String返回值替代

    • ModelMap

      ModelMap是Model接口的实现类,也可以通过ModelMap向页面传递数据,使用Model和ModelMap的效果一样,如果直接使用Model,springmvc会实例化ModelMap

配置

  • web.xml

    <web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
             xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
             xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee                                   http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd"
             version="3.1">
        <!-- 指定Spring 配置文件位置 -->
        <context-param>
            <param-name>contextConfigLocation</param-name>
            <param-value>classpath:applicationContext.xml</param-value>
        </context-param>
        <!-- Spring随服务器启动而启动 -->
        <listener>
            <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
        </listener>
        <!--spring mvc 前端控制器-->
        <servlet>
            <servlet-name>dispatcherServlet</servlet-name>
            <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
            <!-- spring MVC配置文件 -->
            <init-param>
                <param-name>contextConfigLocation</param-name>
                <param-value>classpath:dispatcher-servlet.xml</param-value>
            </init-param>
            <load-on-startup>1</load-on-startup>
        </servlet>
        <servlet-mapping>
            <servlet-name>dispatcherServlet</servlet-name>
            <!-- /* 拦截所有jsp js png css 全部拦截 -->
            <!-- *.action *.do 拦截已action,do结尾的请求 -->
            <!-- /* 拦截所有(不包含jsp)(包含js png css) -->
            <url-pattern>*.do</url-pattern>
        </servlet-mapping>
    </web-app>
    
  • dispatcher-servlet.xml

    <!-- 组件扫描器,自动扫描注解组件 -->
    <!-- 配置扫描包,多个包之间用,分隔 -->
    <context:component-scan base-package="com.lirui.springmvc"/>
    
    <!-- 处理器映射器 -->
    <!--<bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping"/>-->
    <!-- 处理器适配器 -->
    <!--<bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter"/>-->
    
    <!-- 注解驱动,代替以上两种方式 -->
    <mvc:annotation-driven />
    
    <!-- 视图解析器 -->
    <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
        <!-- 配置逻辑视图的前缀 -->
        <property name="prefix" value="/WEB-INF/jsp/"/>
        <!-- 配置逻辑视图的后缀 -->
        <property name="suffix" value=".jsp"/>
        <!-- 前缀+逻辑视图名+后缀 -->
    </bean>
    
  • 定义controller

    @Controller
    public class Test1Controller {
        @RequestMapping("/test1")
        public ModelAndView getList() {
            ModelAndView mav = new ModelAndView();
            mav.addObject("key", request域中的数据);
            mav.setViewName("视图路径");
            return mav;
        }
    

Handler支持的参数类型

  • 默认支持的参数类型

    • HttpServletRequest

      通过request对象获取请求信息

    • HttpServletResponse

      通过response处理响应信息

    • HttpSession

      通过session对象得到session中存放的对象

  • 绑定简单类型

    当请求的参数名称和处理器形参名称一致时会将请求参数与形参进行绑定,可以使用@RequestParam处理简单类型的绑定

    @RequestParam(value = "name",required = true,defaultValue = "value")

    ​ value:参数名称,即入参的请求参数名字

    ​ required :是否必须,默认是true,表示请求中一定要有相应的参数,否则将报错

    ​ defaultValue :默认值

  • 绑定pojo对象

    如果提交的参数很多,或者提交的表单中的内容很多的时候,可以使用简单类型接受数据,也可以使用pojo接收数据。pojo对象中的属性名和表单中input的name属性一致

  • 绑定包装pojo对象

  • 自定义参数绑定

    实现参数格式的修改,例如由于日期数据有很多种格式,springmvc没办法把字符串转换成日期类型。所以需要自定义参数绑定

    • 自定义Converter

      实现Converter接口

    • 配置

      方式一

      <!-- 配置注解驱动 -->
      <!-- 如果配置此标签,可以不用配置... -->
      <mvc:annotation-driven conversion-service="conversionService" />
      
      <!-- 转换器配置 -->
      <bean id="conversionService" class="org.springframework.format.support.FormattingConversionServiceFactoryBean">
          <property name="converters">
              <set>
                  <bean class="cn.itcast.springmvc.converter.DateConverter" />
              </set>
          </property>
      </bean>
      

      方式二

      <!--注解适配器 -->
      <bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter">
          <property name="webBindingInitializer" ref="customBinder"></property>
      </bean>
      
      <!-- 自定义webBinder -->
      <bean id="customBinder" class="org.springframework.web.bind.support.ConfigurableWebBindingInitializer">
          <property name="conversionService" ref="conversionService" />
      </bean>
      
      <!-- 转换器配置 -->
      <bean id="conversionService" class="org.springframework.format.support.FormattingConversionServiceFactoryBean">
          <property name="converters">
              <set>
                  <bean class="cn.itcast.springmvc.convert.DateConverter" />
              </set>
          </property>
      </bean>
      //此方法需要独立配置处理器映射器、适配器,不再使用<mvc:annotation-driven/>
      
  • 高级参数绑定

    • 数组

      提交的数组,可以直接使用数组接收,或者pojo的数组属性接收。两种方式任选其一即可

    • list

      jsp页面name属性必须是list属性名+下标+元素属性

      接收List类型的数据必须是pojo的属性,如果方法的形参为ArrayList类型无法正确接收到数据

@RequestMapping()注解

  • URL路径映射

    @RequestMapping(value="xxx")或@RequestMapping("/xxx"),value的值是数组,可以将多个url映射到同一个方法,例如:@RequestMapping(value = { "xxx", "xxxx" })

  • 添加在类上面

    在class上添加@RequestMapping(url)指定通用请求前缀, 限制此类下的所有方法请求url必须以请求前缀开头

  • 请求方法限定

    限定GET方法:@RequestMapping(method =RequestMethod.GET)

    限定POST方法:@RequestMapping(method =RequestMethod.POST)

    GET和POST都可以:@RequestMapping(method = {RequestMethod.GET,RequestMethod.POST})

Controller方法返回值

  • ModelAndView

    controller方法中定义ModelAndView对象并返回,对象中可添加model数据、指定view

  • Void

    在Controller方法形参上可以定义request和response,使用request或response指定响应结果

    • 使用request转发页面

      request.getRequestDispatcher("页面路径").forward(request,response);

    • 可以通过response页面重定向:

      response.sendRedirect("url")

    • 可以通过response指定响应结果,例如响应json数据如下:

      response.getWriter().print("{\"abc\":123}");

  • String

    controller方法返回字符串可以指定逻辑视图名,通过视图解析器解析为物理视图地址

    • Redirect重定向

      return "redirect:xxx"

    • forward转发

      return "forward:xxx"

异常处理

​ 系统中异常包括两类:预期异常和运行时异常RuntimeException,前者通过捕获异常从而获取异常信息,后者主要通过规范代码开发、测试通过手段减少运行时异常的发生。

​ 系统的dao、service、controller出现都通过throws Exception向上抛出,最后由springmvc前端控制器交由异常处理器(HandlerExceptionResolver)进行异常处理。

  • 自定义异常处理器

    实现接口HandlerExceptionResolver

    public class CustomHandlerExceptionResolver implements HandlerExceptionResolver {
        @Nullable
        @Override
        public ModelAndView resolveException(HttpServletRequest httpServletRequest,                   HttpServletResponse httpServletResponse, @Nullable Object o, Exception e) {
            String msg;
            if (e instanceof CustomException) {
                msg = ((CustomException) e).getMsg();
            } else {
                msg = e.getMessage();
            }
            ModelAndView mav = new ModelAndView();
            mav.addObject("error", msg);
            mav.setViewName("error");
            return mav;
        }
    }
    
  • 配置

    //dispatcher-servlet.xml文件
    <!-- 配置异常处理器 -->
    <bean class="com.lirui.springmvc.exeception.CustomHandlerExceptionResolver"/>
    

上传图片

  • 配置虚拟目录

    在tomcat上配置图片虚拟目录,在tomcat下conf/server.xml中添加:

    <Context docBase="D:\develop\upload\temp"path="/pic" reloadable="false"/>

    访问http://localhost:8080/pic即可访问D:\develop\upload\temp下的图片。也可以通过

    eclipse配置

  • 加入jar文件

    commons-fileupload.jar commons-io.jar

  • 修改jsp页面,添加enctype="multipart/form-data"

  • 采用参数MultipartFile pictureFile接收文件,pictureFile名称与jsp文件上传控件name名称一致

    String fileName = UUID.randomUUID().toString().replace("-", "");
    String pictureName = pictureFile.getOriginalFilename();
    String type = FilenameUtils.getExtension(pictureName);
    fileName = fileName + "." + type;
    pictureFile.transferTo(new File("E:\\Learn\\pic\\" + fileName));
    
  • 配置文件上传解析器

    //dispatcher-servlet.xml
    <!-- 文件上传,id必须设置为multipartResolver -->
    <bean id="multipartResolver"
          class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
        <!-- 设置文件上传大小 -->
        <property name="maxUploadSize" value="5000000" />
    </bean>
    

Json解析

  • 注解

    @RequestBody ---- 用于读取http请求的内容(字符串),通过springmvc提供的HttpMessageConverter接口将读到的内容(json数据)转换为java对象并绑定到Controller方法的参数上

    @ResponseBody ---- 用于将Controller的方法返回的对象,通过springmvc提供HttpMessageConverter接口转换为指定格式的数据如:json,xml等,通过Response响应给客户端

  • 添加json解析器

    如果不使用注解驱动<mvc:annotation-driven/>,就需要给处理器适配器配置json转换器

    在springmvc.xml配置文件中,给处理器适配器加入json转换器:

    <!--处理器适配器 -->
    <bean class=*"org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter"*>
      <property name="messageConverters">
          <list>
              <bean                                       class=*"org.springframework.http.converter.json.MappingJacksonHttpMessageConverter"*>
              </bean>
              </list>
      </property>
    </bean>
    

RESTful支持

使用注解@RequestMapping("item/{id}")声明请求的url,{xxx}叫做占位符,请求的URL可以是“item /1”或“item/2”

使用(@PathVariable() Integer id)获取url上的数据

如果@RequestMapping中表示为"item/{id}",id和形参名称一致,@PathVariable不用指定名称。如果不一致,例如"item/{ItemId}"则需要指定名称@PathVariable("itemId")。

  1. @PathVariable是获取url上数据的。@RequestParam获取请求参数的(包括post表单提交)
  1. 如果加上@ResponseBody注解,就不会走视图解析器,不会返回页面,目前返回的json数据。如果不加,就走视图解析器,返回页面

拦截器

Spring Web MVC 的处理器拦截器类似于Servlet 开发中的过滤器Filter,用于对处理器进行预处理和后处理

  • 拦截器定义(实现HandlerInterceptor接口)

    public class HandlerInterceptor1 implements HandlerInterceptor {
      // controller执行后且视图返回后调用此方法
      // 这里可得到执行controller时的异常信息
      // 这里可记录操作日志
      @Override
      public void afterCompletion(HttpServletRequest arg0, HttpServletResponse arg1, Object           arg2, Exception arg3) throws Exception {}
    
      // controller执行后但未返回视图前调用此方法
      // 这里可在返回用户前对模型数据进行加工处理,比如这里加入公用信息以便页面显示
      @Override
      public void postHandle(HttpServletRequest arg0, HttpServletResponse arg1, Object arg2,          ModelAndView arg3) throws Exception {}
    
      // Controller执行前调用此方法
      // 返回true表示继续执行,返回false中止执行
      // 这里可以加入登录校验、权限拦截等
      @Override
      public boolean preHandle(HttpServletRequest arg0, HttpServletResponse arg1, Object arg2)        throws Exception {
              return true;
          }
    }
    
    

  • 拦截器配置

    <!-- 配置拦截器 -->
    <mvc:interceptors>
      <mvc:interceptor>
          <!-- 所有的请求都进入拦截器 -->
          <mvc:mapping path="/**" />
          <!-- 配置具体的拦截器 -->
          <bean class="xxx" />
      </mvc:interceptor>
      <mvc:interceptor>
          <!-- 所有的请求都进入拦截器 -->
          <mvc:mapping path="/**" />
          <!-- 配置具体的拦截器 -->
          <bean class="xxx" />
      </mvc:interceptor>
    </mvc:interceptors>
    
  • 拦截器处理流程

    preHandle按拦截器定义顺序调用

    postHandler按拦截器定义逆序调用

    afterCompletion按拦截器定义逆序调用

    postHandler在拦截器链内所有拦截器返成功调用

    afterCompletion只有preHandle返回true才调用

springmvc与struts2不同

  • springmvc的入口是一个servlet即前端控制器,而struts2入口是一个filter过滤器。
  • springmvc是基于方法开发(一个url对应一个方法),请求参数传递到方法的形参,可以设计为单例或多例(建议单例),struts2是基于类开发,传递参数是通过类的属性,只能设计为多例。
  • Struts采用值栈存储请求和响应的数据,通过OGNL存取数据, springmvc通过参数解析器是将request请求内容解析,并给方法形参赋值,将数据和视图封装成ModelAndView对象,最后又将ModelAndView中的模型数据通过request域传输到页面。Jsp视图解析器默认使用jstl
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,214评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,307评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,543评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,221评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,224评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,007评论 1 284
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,313评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,956评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,441评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,925评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,018评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,685评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,234评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,240评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,464评论 1 261
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,467评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,762评论 2 345

推荐阅读更多精彩内容