Spring18_SpringMVC的组件解析3

一、SpringMVC的执行流程

 

 1. 用户发送请求至前端控制器DispatcherServlet。

 2. DispatcherServlet收到请求调用HandlerMapping处理器映射器。

 3. 处理器映射器找到具体的处理器(可以根据xml配置、注解进行查找),生成处理器对象及处理器拦截器(如果有则生成)一并返回给DispatcherServlet。

 4. DispatcherServlet调用HandlerAdapter处理器适配器。

 5. HandlerAdapter经过适配调用具体的处理器(Controller,也叫后端控制器)。

 6. Controller执行完成返回ModelAndView。

 7. HandlerAdapter将controller执行结果ModelAndView返回给DispatcherServlet。

 8. DispatcherServlet将ModelAndView传给ViewResolver视图解析器。

 9. ViewReslover解析后返回具体View。

 10. DispatcherServlet根据View进行渲染视图(即将模型数据填充至视图中)。

 11. DispatcherServlet响应用户。

二、SpringMVC组件解析

  1. 前端控制器:DispatcherServlet

  用户请求到达前端控制器,它就相当于MVC模式中的C,DispatcherServlet是整个流程控制的中心,由它调用其它组件处理用户的请求,DispatcherServlet的存在降低了组件之间的耦合性。

 2. 处理器映射器:HandlerMapping
  HandlerMapping负责根据用户请求找到Handler即处理器,SpringMVC提供了不同的映射器实现不同的映射方式,例如:配置文件方式,实现接口方式,注解方式等。 

 3. 处理器适配器:HandlerAdapter

  通过HandlerAdapter对处理器进行执行,这是适配器模式的应用,通过扩展适配器可以对更多类型的处理器进行执行。

 4. 处理器:Handler
  它就是我们开发中要编写的具体业务控制器。由DispatcherServlet把用户请求转发到Handler。由Handler对具体的用户请求进行处理。

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

 6. 视图:View

  SpringMVC框架提供了很多的View视图类型的支持,包括:jstlView、freemarkerView、pdfView等。最常用的视图就是jsp。一般情况下需要通过页面标签或页面模版技术将模型数据通过页面展示给用户,需要由程序员根据业务需求开发具体的页面

三、SpringMVC注解解析

 1. @RequestMapping

  作用:用于建立请求URL和处理请求方法之间的对应关系

  位置:

    • 类上,请求URL 的第一级访问目录。此处不写的话,就相当于应用的根目录
    • 方法上,请求 URL 的第二级访问目录,与类上的使用@ReqquestMapping标注的一级目录一起组成访问虚拟路径

  属性:

    value:用于指定请求的URL。它和path属性的作用是一样的

    method:用于指定请求的方式

    params:用于指定限制请求参数的条件。它支持简单的表达式。要求请求参数的key和value必须和配置的一模一样

       例如:params = {"accountName"},表示请求参数必须有accountName

          params = {"moeny!100"},表示请求参数中money不能是100

  代码实现1:

package com.itheima.controller;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;

@Controller
@RequestMapping("/user")
public class UserController {

    @RequestMapping("/quick")
    public String save(){
        System.out.println("Controller save running...");
        //返回跳转的视图,相对当前资源quick所在的地址http://localhost:8080/user/
        return "success.jsp";
    }
}

   启动tomcat,访问http://localhost:8080/user/quick

    

   报错是因为success.jsp的地址是相对当前资源quick所在的地址,即http://localhost:8080/user/success.jsp,需要改成“/success.jsp”,表示从当前web应用下找资源,重启tomcat,访问http://localhost:8080/user/quick

      

  代码实现2:

package com.itheima.controller;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;

@Controller
@RequestMapping("/user")
public class UserController {

    @RequestMapping(value="/quick",method = RequestMethod.POST)
    public String save(){
        System.out.println("Controller save running...");
        return "/success.jsp";
    }
}

   启动tomcat,访问http://localhost:8080/user/quick,检查结果:

    

   报错提示请求GET方法不被支持,改成method = RequestMethod.GET后,再次启动tomcat,访问http://localhost:8080/user/quick成功

  代码实现3:

package com.itheima.controller;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;

@Controller
@RequestMapping("/user")
public class UserController {

    @RequestMapping(value="/quick",method = RequestMethod.GET,params = {"username"})
    public String save(){
        System.out.println("Controller save running...");
        return "/success.jsp";
    }
}

   启动tomcat,访问http://localhost:8080/user/quick,检查结果:

    

   访问http://localhost:8080/user/quick?username=xxx,则访问成功。

 2. mvc命名空间引入

命名空间:xmlns:context="http://www.springframework.org/schema/context"   
              xmlns:mvc="http://www.springframework.org/schema/mvc"
约束地址:http://www.springframework.org/schema/context     
             http://www.springframework.org/schema/context/spring-context.xsd 
             http://www.springframework.org/schema/mvc 
             http://www.springframework.org/schema/mvc/spring-mvc.xsd

 3. 组件扫描

  SpringMVC基于Spring容器,所以在进行SpringMVC操作时,需要将Controller存储到Spring容器中,如果使用@Controller注解标注的话,就需要使用<context:component-scan base-package=“com.itheima.controller"/>进行组件扫描。

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
                           http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd">

    <!--Controller的组件扫描-->
    <!--如果配置com.itheima则和spring扫重复了,所以需要指定具体扫描哪个注解-->
<!--    <context:component-scan base-package="com.itheima.controller"/>-->
    <context:component-scan base-package="com.itheima">
        <!--表示只扫描com.itheima包下的Controller注解->
        <context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
    </context:component-scan>

</beans>

四、SpringMVC的XML配置解析

 1. SpringMVC有默认组件配置,默认组件都是DispatcherServlet.properties配置文件中配置的,该配置文件地址spring-webmvc.jar包下org/springframework/web/servlet/DispatcherServlet.properties

  

  HandlerMapping:处理器映射器

  HandlerAdapter:处理器适配器

  ViewResolver:视图解析器

 2. 视图解析器: 

org.springframework.web.servlet.ViewResolver=org.springframework.web.servlet.view.InternalResourceViewResolver

  翻看该解析器源码,可以看到该解析器的默认设置,如下:

  

  

REDIRECT_URL_PREFIX = "redirect:"                 --重定向地址的前缀 
FORWARD_URL_PREFIX = "forward:"                   --转发地址的前缀(默认值) 
prefix = "";                                      --视图名称前缀
suffix = "";                                      --视图名称后缀

  2.1 重定向

   浏览器访问http://localhost:8080/user/quick?username=xxx可以看到地址没变,则是转发行为,如果地址变了就是重定向。

package com.itheima.controller;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;

@Controller
@RequestMapping("/user")
public class UserController {

    @RequestMapping(value="/quick",method = RequestMethod.GET,params = {"username"})
    public String save(){
        System.out.println("Controller save running...");
        //"/success.jsp"相当于"forward:/success.jsp"
        return "redirect:/success.jsp";
    }
}

  浏览器访问http://localhost:8080/user/quick?username=xxx,可以看到重定向后地址变了。

   

  2.2 我们可以通过属性注入的方式修改视图的前后

    

   启动tomcat,浏览器访问localhost:8080/user/quick?username=xxx,可以看到视图解析器转发请求到jsp下的success.jsp文件了。

    

   这样写有个弊端,每次资源请求或转发,前面都得加/jsp目录,且文件名都得带.jsp。可以在spring-mvc.xml中通过属性注入的方式去配置解决:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
                           http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd">

    <!--Controller的组件扫描-->
    <context:component-scan base-package="com.itheima.controller"/>

    <!--配置内部资源视图解析器-->
    <bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver">
        <!-- /jsp/success.jsp -->
        <property name="prefix" value="/jsp/"/>
        <property name="suffix" value=".jsp"/>
    </bean>

</beans>
package com.itheima.controller;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;

@Controller
@RequestMapping("/user")
public class UserController {

    @RequestMapping(value="/quick",method = RequestMethod.GET,params = {"username"})
    public String save(){
        System.out.println("Controller save running...");
        return "success";
    }
}

   启动tomcat,浏览器访问localhost:8080/user/quick?username=xxx,可以看到视图解析器最终转发请求到jsp下的success.jsp文件了。

    

五、知识要点

 SpringMVC的相关组件

  • 前端控制器:DispatcherServlet
  • 处理器映射器:HandlerMapping
  • 处理器适配器:HandlerAdapter
  • 处理器:Handler
  • 视图解析器:ViewResolver
  • 视图:View

 SpringMVC的注解和配置 

  • 请求映射注解:@RequestMapping
  • 视图解析器配置:

    REDIRECT_URL_PREFIX = "redirect:" 

    FORWARD_URL_PREFIX = "forward:" 

    prefix = ""; 

    suffix = "";

 

posted on 2023-05-10 12:04  花溪月影  阅读(18)  评论(0编辑  收藏  举报