SpringMVC工作流程
SpringMVC工作流程
CAMELLIA工作流程解析
1. 初始化及单次请求过程
为了更好的使用SpringMVC,将SpringMVC的使用过程总共分两个阶段来分析,分别是启动服务器初始化过程
和单次请求过程
1.1 启动服务器初始化过程
服务器启动,执行ServletContainersInitConfig类,初始化web容器。
- 功能类似于以前的web.xml
执行createServletApplicationContext方法,创建了WebApplicationContext对象。
- 该方法加载SpringMVC的配置类SpringMvcConfig来初始化SpringMVC的容器
加载SpringMvcConfig配置类
执行@ComponentScan加载对应的bean
- 扫描指定包及其子包下所有类上的注解,如Controller类上的@Controller注解
加载UserController,每个@RequestMapping的名称对应一个具体的方法
- 此时就建立了
/save
和 save方法的对应关系
- 此时就建立了
执行getServletMappings方法,设定SpringMVC拦截请求的路径规则
/
代表所拦截请求的路径规则,只有被拦截后才能交给SpringMVC来处理请求
1.2 单次请求过程
发送请求
http://localhost/save
web容器发现该请求满足SpringMVC拦截规则,将请求交给SpringMVC处理
解析请求路径/save
由/save匹配执行对应的方法save()
- 上面的第五步已经将请求路径和方法建立了对应关系,通过/save就能找到对应的save方法
执行save()
检测到有@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,如:
从包结构来看的话,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配置类
public class SpringMvcConfig {
}
// Spring配置类
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//控制层
public class UserController {
public String save(){
System.out.println("user save ...");
return "{'info':'springmvc'}";
}
}
//业务层
public interface UserService {
public void save(User user);
}
public class UserServiceImpl implements UserService {
public void save(User user) {
System.out.println("user service ...");
}
}
//数据层
public interface UserDao {
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 |
|
说明:
上述只是通过例子说明可以精确指定让Spring扫描对应的包结构,真正在做开发的时候,因为Dao最终是交给MapperScannerConfigurer
对象来进行扫描处理的,我们只需要将其扫描到service包即可。
- 方式二:修改Spring配置类,设定扫描范围为com.itheima,排除掉controller包中的bean
1 |
|
excludeFilters属性:设置扫描加载bean时,排除的过滤规则
type属性:设置排除规则,当前使用按照bean定义时的注解类型进行排除
- ANNOTATION:按照注解排除
- ASSIGNABLE_TYPE:按照指定的类型过滤
- ASPECTJ:按照Aspectj表达式排除,基本上不会用
- REGEX:按照正则表达式排除
- CUSTOM:按照自定义规则排除
classes属性:设置排除的具体注解类,当前设置排除@Controller定义的bean
测试controller类是否被排除:
1 | public class App{ |
如果被排除了,该方法执行就会报bean未被定义的错误
==注意:测试的时候,需要把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 | public class ServletContainersInitConfig extends AbstractDispatcherServletInitializer { |
对于上述的配置方式,Spring还提供了一种更简单的配置方式,可以不用再去创建AnnotationConfigWebApplicationContext
对象,不用手动register
对应的配置类。
1 | public class ServletContainersInitConfig extends AbstractAnnotationConfigDispatcherServletInitializer { |
知识点1:@ComponentScan
名称 | @ComponentScan |
---|---|
类型 | 类注解 |
位置 | 类定义上方 |
作用 | 设置spring配置类扫描路径,用于加载使用注解格式定义的bean |
相关属性 | excludeFilters:排除扫描路径中加载的bean,需要指定类别(type)和具体项(classes) includeFilters:加载指定的bean,需要指定类别(type)和具体项(classes) |