2021-11-02昨日复习DispatcherServlet的源码分析,Spring和Spring MVC的整合,修改xml文件进行加载,两个容器的区别,基础理论题

1、DispatcherServlet的源码分析

在这里插入图片描述

在DispatcherServlet类中,有一个初始化策略方法initStrategies方法
在这里插入图片描述
init(初始化)Multipart(文件上传)Resolver(解析器)

protected void initStrategies(ApplicationContext context) {
		initMultipartResolver(context);//初始化文件上传解析
		initLocaleResolver(context);//本地化解析
		initThemeResolver(context);
		initHandlerMappings(context);//通过HandlerMapping,将请求映射到处理器
		initHandlerAdapters(context);//通过HandlerAdapters支持多种类型的解析器
		initHandlerExceptionResolvers(context);//处理错误解析器
		initRequestToViewNameTranslator(context);
		initViewResolvers(context);//视图解析器
		initFlashMapManager(context);
	}

分别可以对应昨天上课DispatcherServlet的用途。
DispatcherServlet 主要用作职责调度工作,本身主要用于控制流程,主要职责如下:

  1. 文件上传解析,如果请求类型是 multipart 将通过 MultipartResolver 进行文件上传
  2. 通过HandlerMapping(处理器映射器),将请求映射到处理器(返回一个HandlerExecutionChain(处理器执行链),包含:一个处理器、多个HandlerInterceptor 拦截器);
  3. 通过 HandlerAdapter处理器适配)支持多种类型的处理器(HandlerExecutionChain 中的处理器);
  4. 通过 ViewResolver视图解析器)解析逻辑视图名到具体视图实现;
  5. 本地化解析; (自动识别本地语言并输出,支持各种语言) 渲染具体的视图等;
  6. 如果执行过程中遇到异常将交给 HandlerExceptionResolver(处理器异常解析器) 来解析。

2、Spring和Spring MVC的整合


SSM:Spring 、Spring MVC 、Mybatis

与昨天的Spring项目的结合。
在resources目录下新建applicationContext.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">
       
 <!-- 排除 com.neu.springmvc包下的所有的 @Controller注解的组件扫描 -->
    <context:component-scan base-package="com.neu.springmvc" use-default-filters="true">
        <context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
    </context:component-scan>
</beans>

在这里,实际上只能扫描到@Service和@Repository注解

在HelloController中加入一条注解
在这里插入图片描述
注意上面的Controller,第一个是注解,第二个是接口,注解不能被实现,一般是用来加载类或者属性、方法上面 。

这个配置文件默认情况下,并不会被自动加载。所以,需要我们在项目的src/main/webapp/WEB-INF/web.xml 中对其进行配置,代码如下:

 <!--    Web项目中加载Spring 的配置-->
    <context-param>
        <param-name>contextConfigLocation</param-name>
        <!--        加载的目录-->
        <param-value>classpath:applicationContext.xml</param-value>
    </context-param>
        <!--    监听器,监听org.springframework.web.context.ContextLoaderListener相关的配置-->
    <listener>
        <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
    </listener>

• contextConfigLocation:表示用于加载 Bean 的配置文件;
• contextClass:表示用于加载Bean的ApplicationContext 实现类,默认WebApplicationContext。

在这里插入图片描述

新建一个HelloController2

package com.neu.springmvc.controller;

import com.neu.springmvc.service.HelloService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.servlet.ModelAndView;
import org.springframework.web.servlet.mvc.Controller;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;


@org.springframework.stereotype.Controller("/hello2")
public class HelloController2 implements Controller {

    @Autowired
    private HelloService helloService;

    public ModelAndView handleRequest(HttpServletRequest req, HttpServletResponse resp) throws Exception {
        ModelAndView mv = new ModelAndView("hello");
        mv.addObject("name", "123");
        return mv;
    } 
} 

为了在 SpringMVC 容器中能够扫描到 HelloController2 ,这里给 HelloController 2添加了 @Controller 注解,同时,由于我们目前采用的 HandlerMapping 是BeanNameUrlHandlerMapping(意味着请求地址就是处理器Bean 的名字), 所以,还需要手动指定 HelloController2 的名字。

之后修改 Spring MVC 的配置文件(spring-servlet.xml),将 Bean 配置为扫描的方式,具体配置代码

<!-- 只扫描com.neu.springmvc及其子包下的包含 @Controller注解的组件  -->
<context:component-scan base-package="com.neu.springmvc" 
                        use-default-filters="false">  
    <context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
</context:component-scan>

如果加入代码报错,说明少了一个配置,在beans xmlns。。。。。中加入

xmlns:context="http://www.springframework.org/schema/context
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd

在这里插入图片描述

配置完成后,再次启动项目,Spring 容器也将会被创建。访问 /hello2 接口, HelloService 中的 hello() 方法就会自动被调用。

之后我们模拟写一个完整的web,模拟dao和service

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述
之后就可以运行了,在地址栏输入http://localhost:8080/hello2?name=王三
在这里插入图片描述

3、Spring和Spring MVC两个容器之间的区别

当Spring和SpringMVC同时出现,我们的项目中将会存在两个容器,一个是Spring 容器,另一个是 Spring MVC 容器。对于Spring 容器,是通过监听器类(ContextLoaderListener)来加载;而对于Spring MVC 容器,则是通过我们配置的 Servlet类(DispatcherServlet)来加载,这两个容器不一样:
在这里插入图片描述
看图可知
1、ContextLoaderListener 初始化的上下文加载的 Bean 是对于整个应用程序共享的,不管是使用什么表现层技术(Spring MVC、Struts2),一般如 :Dao 层、Service 层 的Bean;

2、DispatcherServlet 初始化的上下文加载的 Bean 是只对 Spring Web MVC 有效的Bean,如: Controller、HandlerMapping、HandlerAdapter 等等,该初始化上下文应该只加载 Web相关组件。

注意,WEB-INF目录下的文件无法访问,虽然文件存在。但是WEB-INF是整个web的核心,所有的重要配置文件都在WEB-INF中,不能被轻易的被外界访问。

4、基础理论题

1、为什么不在 Spring容器中扫描所有 Bean?

是不可能的。因为请求达到服务端后,找Spring MVC 中的DispatcherServlet去处理,只会去 Spring MVC容器中找,这就意味着 Controller 必须在 Spring MVC 容器中注入扫描。

2.为什么不在Spring MVC容器中扫描所有Bean?

是可以的,可以在 Spring MVC 容器中扫描所有Bean。之所有不写在一起,有两 个方面的原因:
1).为了方便配置文件的管理
2).在 SSM(Spring+SpringMVC+MyBatis)的组合中,实际上也不支持这种写法。

1.Web层的配置一定要写在Spring MVC配置文件中。
2.非Web层的配置一定要写在Spring配置文件中吗?
答:不一定,也可以写在Spring MVC的配置文件中。

5、处理器详解

HandlerMapping
注意:下文所说的处理器Handler,即我们平时所指的控制器 Controller。

HandlerMapping ,中文译作:处理器映射器。是负责根据 request 请求找到对应的Handler处理器及 Interceptor拦截器,将它们封装在HandlerExecutionChain(处理器执行链) 对象中返回给前端控制器(DispatcherServlet)。
下面列举几个处理器作为说明

•BeanNameUrlHandlerMapping

BeanNameUrl 处理器映射器,根据请求的 url (/hello)与 Spring 容器中定义的 bean 的 name 进行匹配,从而从 Spring 容器中找到 bean 实例,就是说,请求的Url 地址就是处理器Bean的名字。

这个 HandlerMapping 配置如下:

<bean class="org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping" id="handlerMapping">  
 </bean>  

• SimpleUrlHandlerMapping

SimpleUrlHandlerMapping 是 BeanNameUrlHandlerMapping 的增强版本,它可以将 url 和处理器 bean 的 id 进行统一映射配置:

<bean id="handlerMapping"
   class="org.springframework.web.servlet.handler.SimpleUrlHandlerMapping">
        <property name="mappings">
            <props>
                <prop key="/test">testController</prop>
                <prop key="/test2">test2</prop>
            </props>
        </property>
    </bean>

注意:在 props 中,可以配置多个请求路径和处理器实例的映射关系。

6、处理器适配器详解

HandlerAdapter,中文译作处理器适配器。 会根据适配器接口后端控制器(HelloController)进行包装(适配),包装后即可对处理器进行执行,通过扩展处理器适配器可以执行多种类型的处理器,这里使用了适配器设计模式。 这个是一个接口。

• SimpleControllerHandlerAdapter

SimpleControllerHandlerAdapter 是简单控制器处理器适配器,所有实现了org.springframework.web.servlet.mvc.Controller 接口的 Bean 通过此适配器进行适配、执行,也就是说,如果我们开发的接口是通过实现 Controller 接口来 完成的(不是通过注解开发的接口),那么 HandlerAdapter 必须是
SimpleControllerHandlerAdapter。

• HttpRequestHandlerAdapter

HttpRequestHandlerAdapter是http 请求处理器适配器,所有实现了
org.springframework.web.HttpRequestHandler 接口的 Bean 通过此适配器进行适配、执行。

整个相关的代码、配置信息如下表:
处理器(Handler)

@Controller("test3")
public class Test3Controller implements HttpRequestHandler {

    @Override
    public void handleRequest(HttpServletRequest request,
                              HttpServletResponse response)
            throws ServletException, IOException {
        System.out.println("-----Test3Controller-----");
        response.getWriter().write("<h1>Test3</h1>");
    }
}

在SpringMVC-servlet.xml 中配置: HttpRequestHandlerAdapter

<!-- HandlerAdapter(处理器适配器) -->
    <!--<bean id="handlerAdapter" class="org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter"/>-->
    <bean id="handlerAdapter"
          class="org.springframework.web.servlet.mvc.HttpRequestHandlerAdapter"/>
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值