SpringMVC入门

回顾之前的JavaWeb的MVC架构

  1. 在web.xml中配置servlet以及映射路径
  2. 在servlet类中获取请求参数并调用相应的业务方法
  3. 最后通过请求转发或者重定向的方式来跳转页面

SpringMVC入门

SpringMVC的请求分发都是依靠一个Servler,DispatcherServlet–前端控制器

开发步骤

  1. 配置前端控制器:DispatcherServlet
  2. 配置处理器映射器:BeanNameUrlHandlerMapping
  3. 配置处理器适配器:SimpleControllerHandlerAdapter
  4. 配置视图解析器:InternalResourceViewResolver
  5. 开发和配置处理器:HelloController

编写Controller

public class HelloController implements Controller {
    public ModelAndView handleRequest(HttpServletRequest request, HttpServletResponse response) throws Exception {
        System.out.println("hello.......");
        ModelAndView mv = new ModelAndView();
        mv.addObject("msg", "你好SpringMVC");

        mv.setViewName("/WEB-INF/views/hello/hello.jsp");
        return mv;
    }
}

同样的,在web.xml中配置好DispatcherServlet后,SpringMVC的前端控制器会继续后面的操作
在xml中配置:

<!-- 底层都是使用autuwired注入,所以有类型即可,不需要id -->
<!--1:处理器映射器-->
<!--
     处理器映射器将处理器(handler)的name作为URL查找
-->
<bean class="org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping"/>

<!--2:处理器适配器-->
<!--
    ①:所有的适配器都需要实现于HandlerAdapter接口
    ②:要求所有的处理器需要实现于Controller接口
-->
<bean class="org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter"/>

<!--3:视图解析器-->
<!--
    缺省使用JSTL
-->
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver"/>

<!--4:处理器-->
<!-- 这里一定是name,因为BeanNameUrlHandlerMapping是根据bean的name来匹配url的 -->
<bean name="/hello" class="com._520it._01_hello.HelloController"/>

首先,通过处理器映射器,找到指定url所对应的bean

// BeanNameUrlHandlerMapping的方法,这里是将以“/”开头的bean都储存好
@Override
protected String[] determineUrlsForHandler(String beanName) {
    List<String> urls = new ArrayList<>();
    if (beanName.startsWith("/")) {
        urls.add(beanName);
    }
    String[] aliases = obtainApplicationContext().getAliases(beanName);
    for (String alias : aliases) {
        if (alias.startsWith("/")) {
            urls.add(alias);
        }
    }
    return StringUtils.toStringArray(urls);
}

接着再通过处理器适配器找到合适的适配器,这里的SimpleControllerHandlerAdapter是找到Controller的适配器,在适配器里有相应的执行方法

// 判断是否合适
@Override
	public boolean supports(Object handler) {
		return (handler instanceof Controller);
	}

// 调用相应的执行方法
@Override
@Nullable
public ModelAndView handle(HttpServletRequest request, HttpServletResponse response, Object handler)
        throws Exception {

    return ((Controller) handler).handleRequest(request, response);
}

执行完后,DispatcherServlet会拿到ModelAndView对象,并且将这个对象交给视图解析器解析
一般的,Spring也使用InternalResourceViewResolver作为视图解析器
在解析中使用render方法来渲染,先获得逻辑视图,再整合数据输出物理视图

DispatcherServlet分析

通过断点,可以找到主要逻辑在doDispatcher方法中,下面将一些主要方法列出来

protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception {
    // 定义HandlerExecutionChain执行链
    HandlerExecutionChain mappedHandler = null;

    // Determine handler for the current request.
    // 通过处理器映射器获得chain执行链,其中还会将interceptor放入执行链中
    mappedHandler = getHandler(processedRequest);

    // Determine handler adapter for the current request.
    // 通过这个执行链获得处理器适配器,(匹配适配器的方法是通过适配器中supports来确定的,如果为false则不匹配)
    HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());

    // 先执行前置拦截
    if (!mappedHandler.applyPreHandle(processedRequest, response)) {
        return;
    }

    // Actually invoke the handler.
    // 调用适配器中的handle方法
    mv = ha.handle(processedRequest, response, mappedHandler.getHandler());

    // 后置拦截等等
    ...

    // 解析视图,底层调用render方法
    processDispatchResult(processedRequest, response, mappedHandler, mv, dispatchException);
}

事实上,在正常开发的SpringMVC中,不需要我们手动配置HandlerMapping、HandlerAdapter,这些在依赖中的DispatcherServlet.properties都已经自动配置好了,而我们开发时也是使用注解进行开发,所以也不需要在xml中配置bean了,而且HandlerMapping用的也是和RequestMapping相关的。而视图解析器的配置主要也是在xml中配置视图的前缀以及后缀等。

DispatcherServlet默认的配置比如HandlerMapping以及HandlerAdapter在DispatcherServlet.properties中配置,在DispatcherServlet中引入

/**
    * Name of the class path resource (relative to the DispatcherServlet class)
    * that defines DispatcherServlet's default strategy names.
    */
private static final String DEFAULT_STRATEGIES_PATH = "DispatcherServlet.properties";

static {
    // Load default strategy implementations from properties file.
    // This is currently strictly internal and not meant to be customized
    // by application developers.
    try {
        ClassPathResource resource = new ClassPathResource(DEFAULT_STRATEGIES_PATH, DispatcherServlet.class);
        defaultStrategies = PropertiesLoaderUtils.loadProperties(resource);
    }
    catch (IOException ex) {
        throw new IllegalStateException("Could not load '" + DEFAULT_STRATEGIES_PATH + "': " + ex.getMessage());
    }
}

补充

SpringMVC对于静态资源访问的处理

在配置前端控制器的写成"/"时,就不能访问到静态资源了,所以有一下解决方式

在SpringMVC的配置文件中添加对静态资源的访问

在xml中加上这一行

<!-- 相当于在mvc中另外指引向服务器(tomcat)的静态资源处理方式 -->
<mvc:default-servlet-handler/>

添加对访问特定路径开头的资源

将静态资源全部放在一个目录下,如static,之后在xml中进行配置

<!-- 表示当访问/static开头的资源时,全都当做静态资源处理 -->
<mvc:resources mapping=”/static/**” location=”/static”/>

url-pattern不要写作"/",改为*.do

此时需要把所有控制器资源的以.do结尾来访问,而静态的资源都不以.do结尾,就会自动交给Tomcat处理
这个方法不适用于RESTFUL规范,RESTFUL规定需要资源路径是"/"

SpringMVC注解的解析器:

<mvc:annotation-driver>

会自动注册RequestMappingHandlerMapping/RequestMappingHandlerAdapter/ExceptionHandlerExceptionResolver三个bean
除此之外,还支持
支持使用 ConversionService实例对表单参数进行类型转换.
支持使用 @NumberFormat、@DateTimeFormat注解完成数据格式化操作.
支持使用 @Valid注解对JavaBean实例进行JSR303验证.
支持使用 @RequestBody和@ResponseBody注解读写JSON.

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值