SpringMVC工作流程

工作流程解析


1. 初始化及单次请求过程


为了更好的使用SpringMVC,将SpringMVC的使用过程总共分两个阶段来分析,分别是启动服务器初始化过程单次请求过程

1630432494752

1.1 启动服务器初始化过程

  1. 服务器启动,执行ServletContainersInitConfig类,初始化web容器。

    • 功能类似于以前的web.xml
  2. 执行createServletApplicationContext方法,创建了WebApplicationContext对象。

    • 该方法加载SpringMVC的配置类SpringMvcConfig来初始化SpringMVC的容器
  3. 加载SpringMvcConfig配置类

  4. 执行@ComponentScan加载对应的bean

    • 扫描指定包及其子包下所有类上的注解,如Controller类上的@Controller注解
  5. 加载UserController,每个@RequestMapping的名称对应一个具体的方法

    • 此时就建立了 /save 和 save方法的对应关系
  6. 执行getServletMappings方法,设定SpringMVC拦截请求的路径规则

    • /代表所拦截请求的路径规则,只有被拦截后才能交给SpringMVC来处理请求

1.2 单次请求过程

  1. 发送请求http://localhost/save

  2. web容器发现该请求满足SpringMVC拦截规则,将请求交给SpringMVC处理

  3. 解析请求路径/save

  4. 由/save匹配执行对应的方法save()

    • 上面的第五步已经将请求路径和方法建立了对应关系,通过/save就能找到对应的save方法
  5. 执行save()

  6. 检测到有@ResponseBody直接将save()方法的返回值作为响应体返回给请求方

2. bean加载控制


2.1 问题分析

  • config目录存入的是配置类,写过的配置类有:

    • ServletContainersInitConfig
    • SpringConfig
    • SpringMvcConfig
    • JdbcConfig
    • MybatisConfig
  • controller目录存放的是SpringMVC的controller类

  • service目录存放的是service接口和实现类

  • dao目录存放的是dao/Mapper接口

controller、service和dao这些类都需要被容器管理成bean对象,那么到底是该让SpringMVC加载还是让Spring加载呢?

  • SpringMVC加载其相关bean(表现层bean),也就是controller包下的类。
  • Spring控制的bean
    • 业务bean(Service)
    • 功能bean(DataSource,SqlSessionFactoryBean,MapperScannerConfigurer等)

分析清楚谁该管哪些bean以后,接下来要解决的问题是如何让Spring和SpringMVC分开加载各自的内容。

在SpringMVC的配置类SpringMvcConfig中使用注解@ComponentScan,只需要将其扫描范围设置到controller即可。

在Spring的配置类SpringConfig中使用注解@ComponentScan,当时扫描的范围中其实是已经包含了controller,如:

1630460408159

从包结构来看的话,Spring已经多把SpringMVC的controller类也给扫描到,==因为功能不同,如何避免Spring错误加载到SpringMVC的bean?==

2.2 思路分析

针对上面的问题,解决方案也比较简单,就是:

  • 加载Spring控制的bean的时候排除掉SpringMVC控制的bean

具体该如何排除:

  • 方式一: Spring加载的bean设定扫描范围为精准范围,例如service包、dao包等
  • 方式二: Spring加载的bean设定扫描范围为com.itheima,排除掉controller包中的bean
  • 方式三: 不区分Spring与SpringMVC的环境,加载到同一个环境中[了解即可]

2.3 环境准备

  • 创建对应的配置类

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    //创建WebApplicationContext对象
    public class ServletContainersInitConfig extends AbstractDispatcherServletInitializer {

    protected WebApplicationContext createServletApplicationContext() {
    AnnotationConfigWebApplicationContext ctx = new AnnotationConfigWebApplicationContext();
    ctx.register(SpringMvcConfig.class);
    return ctx;
    }

    protected String[] getServletMappings() {
    return new String[]{"/"};
    }

    protected WebApplicationContext createRootApplicationContext() {
    return null;
    }

    }

    //SpringMVC配置类
    @Configuration
    @ComponentScan("com.itheima.controller")
    public class SpringMvcConfig {
    }

    // Spring配置类
    @Configuration
    @ComponentScan("com.itheima")
    public class SpringConfig {
    }

  • 编写Controller,Service,Dao,Domain类

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    //控制层
    @Controller
    public class UserController {
    @RequestMapping("/save")
    @ResponseBody
    public String save(){
    System.out.println("user save ...");
    return "{'info':'springmvc'}";
    }
    }

    //业务层
    public interface UserService {
    public void save(User user);
    }

    @Service
    public class UserServiceImpl implements UserService {
    public void save(User user) {
    System.out.println("user service ...");
    }
    }

    //数据层
    public interface UserDao {
    @Insert("insert into tbl_user(name,age)values(#{name},#{age})")
    public void save(User user);
    }
    public class User {
    private Integer id;
    private String name;
    private Integer age;
    //setter..getter..toString略
    }

2.4 *设置bean加载控制

  • 方式一:修改Spring配置类,设定扫描范围为精准范围。
1
2
3
4
@Configuration
@ComponentScan({"com.itheima.service","comitheima.dao"})
public class SpringConfig {
}

说明:

上述只是通过例子说明可以精确指定让Spring扫描对应的包结构,真正在做开发的时候,因为Dao最终是交给MapperScannerConfigurer对象来进行扫描处理的,我们只需要将其扫描到service包即可。

  • 方式二:修改Spring配置类,设定扫描范围为com.itheima,排除掉controller包中的bean
1
2
3
4
5
6
7
8
9
@Configuration
@ComponentScan(value="com.itheima",
excludeFilters=@ComponentScan.Filter(
type = FilterType.ANNOTATION,
classes = Controller.class
)
)
public class SpringConfig {
}
  • excludeFilters属性:设置扫描加载bean时,排除的过滤规则

  • type属性:设置排除规则,当前使用按照bean定义时的注解类型进行排除

    • ANNOTATION:按照注解排除
    • ASSIGNABLE_TYPE:按照指定的类型过滤
    • ASPECTJ:按照Aspectj表达式排除,基本上不会用
    • REGEX:按照正则表达式排除
    • CUSTOM:按照自定义规则排除
  • classes属性:设置排除的具体注解类,当前设置排除@Controller定义的bean

测试controller类是否被排除:

1
2
3
4
5
6
public class App{
public static void main (String[] args){
AnnotationConfigApplicationContext ctx = new AnnotationConfigApplicationContext(SpringConfig.class);
System.out.println(ctx.getBean(UserController.class));
}
}

如果被排除了,该方法执行就会报bean未被定义的错误

1630462200947

==注意:测试的时候,需要把SpringMvcConfig配置类上的@ComponentScan注解注释掉,否则不会报错。==

  • Spring配置类扫描的包是com.itheima

    • 排除掉controller包中的bean,所以不会扫描controller包下的bean。
    • Spring的配置类可以正常扫描SpringMvcConfig
  • SpringMVC的配置类,SpringMvcConfig上有一个@Configuration注解。

    • SpringMvcConfig设置的扫描范围是controller包下。
  • 所以,Spring的配置类可以通过SpringMvcConfig将controller包下的bean加载到IOC容器。

    可以把SpringMVC的配置类移出Spring配置类的扫描范围(放置在其他包)

有了Spring的配置类,要想在tomcat服务器启动将其加载,我们需要修改ServletContainersInitConfig

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
public class ServletContainersInitConfig extends AbstractDispatcherServletInitializer {
protected WebApplicationContext createServletApplicationContext() {
AnnotationConfigWebApplicationContext ctx = new AnnotationConfigWebApplicationContext();
ctx.register(SpringMvcConfig.class);
return ctx;
}
protected String[] getServletMappings() {
return new String[]{"/"};
}

//初始化 Spring 的根应用上下文
protected WebApplicationContext createRootApplicationContext() {
AnnotationConfigWebApplicationContext ctx = new AnnotationConfigWebApplicationContext();
ctx.register(SpringConfig.class);
return ctx;
}
}

对于上述的配置方式,Spring还提供了一种更简单的配置方式,可以不用再去创建AnnotationConfigWebApplicationContext对象,不用手动register对应的配置类。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
public class ServletContainersInitConfig extends AbstractAnnotationConfigDispatcherServletInitializer {

protected Class<?>[] getRootConfigClasses() {
return new Class[]{SpringConfig.class}; //写上Spring的配置类即可
}

protected Class<?>[] getServletConfigClasses() {
return new Class[]{SpringMvcConfig.class}; //写上SpringMVC的配置类即可
}

protected String[] getServletMappings() {
return new String[]{"/"};
}
}

知识点1:@ComponentScan

名称 @ComponentScan
类型 类注解
位置 类定义上方
作用 设置spring配置类扫描路径,用于加载使用注解格式定义的bean
相关属性 excludeFilters:排除扫描路径中加载的bean,需要指定类别(type)和具体项(classes)
includeFilters:加载指定的bean,需要指定类别(type)和具体项(classes)