在web.xml中溶浴,可以找到這么一段配置文件荆几,它的作用是將DispatchServlet作為一個(gè)servlet加載到servlet容器中
其中粗井,<url-pattern>是將某一類(lèi)請(qǐng)求轉(zhuǎn)交給這個(gè)servlet進(jìn)行處理
<servlet>
<servlet-name>SpringMVC</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:spring-mvc.xml</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
//是否開(kāi)啟異步支持
<!--<async-supported>true</async-supported>-->
</servlet>
<servlet-mapping>
<servlet-name>SpringMVC</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
然后我們開(kāi)始解析一下DispatchServlet具體做了些什么事情
1)加載DispatcherServlet.properties中的屬性
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 'DispatcherServlet.properties': " + ex.getMessage());
}
}
找到DispatcherServlet.properties唧取,其中內(nèi)容如下
# Default implementation classes for DispatcherServlet's strategy interfaces.
# Used as fallback when no matching beans are found in the DispatcherServlet context.
# Not meant to be customized by application developers.
org.springframework.web.servlet.LocaleResolver=org.springframework.web.servlet.i18n.AcceptHeaderLocaleResolver
org.springframework.web.servlet.ThemeResolver=org.springframework.web.servlet.theme.FixedThemeResolver
org.springframework.web.servlet.HandlerMapping=org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping,\
org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping
org.springframework.web.servlet.HandlerAdapter=org.springframework.web.servlet.mvc.HttpRequestHandlerAdapter,\
org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter,\
org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter
org.springframework.web.servlet.HandlerExceptionResolver=org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerExceptionResolver,\
org.springframework.web.servlet.mvc.annotation.ResponseStatusExceptionResolver,\
org.springframework.web.servlet.mvc.support.DefaultHandlerExceptionResolver
org.springframework.web.servlet.RequestToViewNameTranslator=org.springframework.web.servlet.view.DefaultRequestToViewNameTranslator
org.springframework.web.servlet.ViewResolver=org.springframework.web.servlet.view.InternalResourceViewResolver
org.springframework.web.servlet.FlashMapManager=org.springframework.web.servlet.support.SessionFlashMapManager
所以我們可以知道航夺,這個(gè)DispatcherServlet.properties的作用是將默認(rèn)實(shí)現(xiàn)類(lèi)的路徑以鍵值對(duì)的形式加載進(jìn)來(lái)
LocaleResolver(本地化解析器媳叨,AcceptHeaderLocaleResolver)
ThemeResolver(主題解析器腥光,F(xiàn)ixedThemeResolver)
HandlerMapping(映射處理器,BeanNameUrlHandlerMapping)
HandlerAdapter(處理適配器糊秆,多個(gè))
ViewResolver(視圖解析器武福,InternalResourceViewResolver)
RequestToViewNameTranslator(將請(qǐng)求名稱(chēng)轉(zhuǎn)化為視圖名稱(chēng),DefaultRequestToViewNameTranslator)
2)初始化加載SpringMVC框架下需要配置的基本的bean
protected void initStrategies(ApplicationContext context) {
initMultipartResolver(context);
initLocaleResolver(context);
initThemeResolver(context);
initHandlerMappings(context);
initHandlerAdapters(context);
initHandlerExceptionResolvers(context);
initRequestToViewNameTranslator(context);
initViewResolvers(context);
initFlashMapManager(context);
}
這里的作用是將SpringMVC框架所需的各個(gè)基本的bean初始化痘番,并加載到容器中
3)重寫(xiě)doService()方法
DispatchServlet實(shí)際上繼承于FrameworkServlet捉片,并且在其具體實(shí)現(xiàn)中重寫(xiě)了doService()方法平痰。
doService()主要是將一些全局屬性set到request中,具體的請(qǐng)求處理則調(diào)用并在doDispatch()中處理
4)doDipatch()將請(qǐng)求處理后轉(zhuǎn)發(fā)到Controller
/**
* Process the actual dispatching to the handler.
* <p>The handler will be obtained by applying the servlet's HandlerMappings in order.
* The HandlerAdapter will be obtained by querying the servlet's installed HandlerAdapters
* to find the first that supports the handler class.
* <p>All HTTP methods are handled by this method. It's up to HandlerAdapters or handlers
* themselves to decide which methods are acceptable.
* @param request current HTTP request
* @param response current HTTP response
* @throws Exception in case of any kind of processing failure
*/
protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception {
//獲取當(dāng)前要處理的request
HttpServletRequest processedRequest = request;
HandlerExecutionChain mappedHandler = null;
boolean multipartRequestParsed = false;
WebAsyncManager asyncManager = WebAsyncUtils.getAsyncManager(request);
try {
ModelAndView mv = null;
Exception dispatchException = null;
try {
//判斷是否有文件上傳
processedRequest = checkMultipart(request);
multipartRequestParsed = (processedRequest != request);
// Determine handler for the current request.
//獲取HandlerExecutionChain伍纫,其中包含了private HandlerInterceptor[] interceptors宗雇,即攔截器的一個(gè)數(shù)組
mappedHandler = getHandler(processedRequest);
if (mappedHandler == null || mappedHandler.getHandler() == null) {
noHandlerFound(processedRequest, response);
return;
}
// Determine handler adapter for the current request.
//獲取HandlerAdapter
HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());
// Process last-modified header, if supported by the handler.
//從request中獲取HTTP請(qǐng)求方法
String method = request.getMethod();
boolean isGet = "GET".equals(method);
if (isGet || "HEAD".equals(method)) {
long lastModified = ha.getLastModified(request, mappedHandler.getHandler());
if (logger.isDebugEnabled()) {
logger.debug("Last-Modified value for [" + getRequestUri(request) + "] is: " + lastModified);
}
if (new ServletWebRequest(request, response).checkNotModified(lastModified) && isGet) {
return;
}
}
//如果有攔截器,則處理其前置方法preHandle
if (!mappedHandler.applyPreHandle(processedRequest, response)) {
return;
}
// Actually invoke the handler.
//返回處理后的modelView
mv = ha.handle(processedRequest, response, mappedHandler.getHandler());
//判斷是否是異步請(qǐng)求
if (asyncManager.isConcurrentHandlingStarted()) {
return;
}
//如果view為空莹规,返回一個(gè)默認(rèn)的view
applyDefaultViewName(processedRequest, mv);
//處理攔截器的后置方法postHandle
mappedHandler.applyPostHandle(processedRequest, response, mv);
}
catch (Exception ex) {
dispatchException = ex;
}
processDispatchResult(processedRequest, response, mappedHandler, mv, dispatchException);
}
catch (Exception ex) {
triggerAfterCompletion(processedRequest, response, mappedHandler, ex);
}
catch (Error err) {
triggerAfterCompletionWithError(processedRequest, response, mappedHandler, err);
}
finally {
//判斷是否是異步請(qǐng)求
if (asyncManager.isConcurrentHandlingStarted()) {
// Instead of postHandle and afterCompletion
if (mappedHandler != null) {
mappedHandler.applyAfterConcurrentHandlingStarted(processedRequest, response);
}
}
else {
// Clean up any resources used by a multipart request.
//刪除上傳資源
if (multipartRequestParsed) {
cleanupMultipart(processedRequest);
}
}
}
}