回顾之前的JavaWeb的MVC架构
- 在web.xml中配置servlet以及映射路径
- 在servlet类中获取请求参数并调用相应的业务方法
- 最后通过请求转发或者重定向的方式来跳转页面
SpringMVC入门
SpringMVC的请求分发都是依靠一个Servler,DispatcherServlet–前端控制器
开发步骤
- 配置前端控制器:DispatcherServlet
- 配置处理器映射器:BeanNameUrlHandlerMapping
- 配置处理器适配器:SimpleControllerHandlerAdapter
- 配置视图解析器:InternalResourceViewResolver
- 开发和配置处理器: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.