- Springmvc source code analysis
- The boot process
- SpringMvc configuration parsing
- DispatcherServlet call process
- HandleMapping
- HandleAdapter
- The DispatcherServlet invokes the HandlerAdapter procedure
SpringMVC source code analysis series:
- SpringMVC source code Parsing (1)- Startup process
- SpringMVC source code parsing (2)-DispatcherServlet
- SpringMVC source code parsing (3)-HandleMapping
- SpringMVC source code parsing (4)-HandlerAdapter
- SpringMVC source parsing (5)-HandlerExceptionResolver
- SpringMVC source Code Parsing (6)- Asynchronous requests
The boot process
The configuration mode is web. XML
<?xml version="1.0" encoding="UTF-8"? >
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://java.sun.com/xml/ns/javaee"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"
id="WebApp_ID" version="3.0">
<display-name>CtrTimeOut</display-name>
<welcome-file-list>
<welcome-file>index.jsp</welcome-file>
</welcome-file-list>
<context-param>
<param-name>contextConfigLocation</param-name># Spring configuration<param-value>classpath:config/spring/applicationContext.xml</param-value>
</context-param>
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<servlet>
<servlet-name>controller</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name># SpringMVC configuration<param-value>classpath:config/spring/spring-controller.xml</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>controller</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
</web-app>
Copy the code
ContextLoadListener creates the WebApplicationContext as the Spring container context
# org. Springframework. Web. Context. ContextLoader / * * * according to the XML configuration to create applicationContext * / public WebApplicationContext initWebApplicationContext(ServletContext servletContext) { ... try { // Store context in local instance variable, To guarantee that // it is available on ServletContext shutdown. If (this.context == null) {if (this.context == null) { this.context = createWebApplicationContext(servletContext); } if (this.context instanceof ConfigurableWebApplicationContext) { ConfigurableWebApplicationContext cwac = (ConfigurableWebApplicationContext) this.context; if (! cwac.isActive()) { ... / / read contextConfigLocation configuration and refresh () configureAndRefreshWebApplicationContext (cwac, servletContext); }} // Set applicationContext to servletContext servletContext.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE, this.context); . return this.context; }}Copy the code
The DispatcherServlet creates the WebApplicationContext as the springMVC context and sets the context created by ContextLoadListener as its parent
DispatcherServlet extends FrameworkServlet #org.springframework.web.servlet.FrameworkServlet @Override protected final void initServletBean() throws ServletException { ... try { this.webApplicationContext = initWebApplicationContext(); initFrameworkServlet(); }... } protected WebApplicationContext initWebApplicationContext() { WebApplicationContext rootContext = WebApplicationContextUtils.getWebApplicationContext(getServletContext()); WebApplicationContext wac = null; . If (wac = = null) {/ / create applicationContext wac = createWebApplicationContext (rootContext); }... return wac; } protected WebApplicationContext createWebApplicationContext(ApplicationContext parent) { //XmlWebApplicationContext Class<? > contextClass = getContextClass(); . / / create the applicationContext ConfigurableWebApplicationContext wac = (ConfigurableWebApplicationContext) BeanUtils.instantiateClass(contextClass); wac.setEnvironment(getEnvironment()); // setParent(applicationContext created in ContextLoadListener) wac.setparent (parent); SetConfigLocation (getContextConfigLocation()); //refresh() configureAndRefreshWebApplicationContext(wac); return wac; }Copy the code
Springmvc’s applicationContext is going to read the configuration file and let’s look at one of the simplest configuration files
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:mvc="http://www.springframework.org/schema/mvc" xmlns:tx="http://www.springframework.org/schema/tx"
xsi:schemaLocation="Http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.0.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.0.xsd http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-3.0.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-3.0.xsd"
default-autowire="byName">The springMVC container scans the path<context:component-scan base-package="com.iflytek.ossp.ctrtimeout.controller"></context:component-scan># spring4 new labels Main is added to the default HandleMappin ViewResolver, HandleAdapter<mvc:annotation-driven />
</beans>
Copy the code
SpringMvc configuration parsing
According to spring’s custom schema resolution mechanism we find it in the following figure
http\://www.springframework.org/schema/mvc=org.springframework.web.servlet.config.MvcNamespaceHandler
Copy the code
You can see that all the tag parsers for MVC are defined here
public class MvcNamespaceHandler extends NamespaceHandlerSupport {
@Override
public void init(a) {
registerBeanDefinitionParser("annotation-driven".new AnnotationDrivenBeanDefinitionParser());
registerBeanDefinitionParser("default-servlet-handler".new DefaultServletHandlerBeanDefinitionParser());
registerBeanDefinitionParser("interceptors".new InterceptorsBeanDefinitionParser());
registerBeanDefinitionParser("resources".new ResourcesBeanDefinitionParser());
registerBeanDefinitionParser("view-controller".new ViewControllerBeanDefinitionParser());
registerBeanDefinitionParser("redirect-view-controller".new ViewControllerBeanDefinitionParser());
registerBeanDefinitionParser("status-controller".new ViewControllerBeanDefinitionParser());
registerBeanDefinitionParser("view-resolvers".new ViewResolversBeanDefinitionParser());
registerBeanDefinitionParser("tiles-configurer".new TilesConfigurerBeanDefinitionParser());
registerBeanDefinitionParser("freemarker-configurer".new FreeMarkerConfigurerBeanDefinitionParser());
registerBeanDefinitionParser("velocity-configurer".new VelocityConfigurerBeanDefinitionParser());
registerBeanDefinitionParser("groovy-configurer".newGroovyMarkupConfigurerBeanDefinitionParser()); }}Copy the code
Look at the AnnotationDrivenBeanDefinitionParser parsers do
The parsing process is a bit more complicated and we can tell from the comments that the following objects will be loaded
DispatcherServlet call process
protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception {...try {
ModelAndView mv = null;
Exception dispatchException = null;
try{ processedRequest = checkMultipart(request); multipartRequestParsed = (processedRequest ! = request);//1. Call handlerMapping to obtain handlerChain
mappedHandler = getHandler(processedRequest);
if (mappedHandler == null || mappedHandler.getHandler() == null) {
noHandlerFound(processedRequest, response);
return;
}
// 2. Get a HandlerAdapter that supports this handler parsingHandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler()); .// 3. Use the HandlerAdapter to complete the handler processing
mv = ha.handle(processedRequest, response, mappedHandler.getHandler());
if (asyncManager.isConcurrentHandlingStarted()) {
return;
}
// 4. View processing (page rendering)
applyDefaultViewName(request, mv);
mappedHandler.applyPostHandle(processedRequest, response, mv);
}
catch(Exception ex) { dispatchException = ex; } processDispatchResult(processedRequest, response, mappedHandler, mv, dispatchException); }... }Copy the code
Process summary:
- Call HandleMapping to get the handler
- Call HandleAdapter to perform the Handle procedure (parameter resolution procedure call)
- Call ViewResolver for view resolution
- Render view
The above pictures are from the Internet
HandleMapping
Definition: Request path – process mapping management
For example, get a handler(your Controller method) that you can process based on the path of your HTTP request.
/** * Interface to be implemented by objects that define a mapping between * requests and handler objects. */
public interface HandlerMapping {
// Obtain the processing chain according to request
HandlerExecutionChain getHandler(HttpServletRequest request) throws Exception;
}
Copy the code
RequestMappingHandlerMapping, for example Let’s look at his inheritance
You can see that there is an InitlizingBean(Spring’s lifecycle interface) that we can start with
#org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping //1. @Override public void afterPropertiesSet() { if (this.useRegisteredSuffixPatternMatch) { this.fileExtensions.addAll(this.contentNegotiationManager.getAllFileExtensions()); } super.afterPropertiesSet(); } //4. @Override protected boolean isHandler(Class<? > beanType) { return ((AnnotationUtils.findAnnotation(beanType, Controller.class) ! = null) || (AnnotationUtils.findAnnotation(beanType, RequestMapping.class) ! = null)); } //6. @Override protected RequestMappingInfo getMappingForMethod(Method method, Class<? > handlerType) { RequestMappingInfo info = null; RequestMapping methodAnnotation = AnnotationUtils.findAnnotation(method, RequestMapping.class); if (methodAnnotation ! = null) {RequestCondition<? > methodCondition = getCustomMethodCondition(method); info = createRequestMappingInfo(methodAnnotation, methodCondition); RequestMapping typeAnnotation = AnnotationUtils.findAnnotation(handlerType, RequestMapping.class); if (typeAnnotation ! = null) { RequestCondition<? > typeCondition = getCustomTypeCondition(handlerType); info = createRequestMappingInfo(typeAnnotation, typeCondition).combine(info); } } return info; } #org.springframework.web.servlet.handler.AbstractHandlerMethodMapping //2. @Override public void afterPropertiesSet() { initHandlerMethods(); } /** * Scan beans in the ApplicationContext, detect and register handler methods. * @see #isHandler(Class) * @see #getMappingForMethod(Method, Class) * @see #handlerMethodsInitialized(Map) */ //3. protected void initHandlerMethods() { if (logger.isDebugEnabled()) { logger.debug("Looking for request mappings in application context: " + getApplicationContext()); } / / obtain all the object from the container type name String [] beanNames = (enclosing detectHandlerMethodsInAncestorContexts? BeanFactoryUtils.beanNamesForTypeIncludingAncestors(getApplicationContext(), Object.class) : getApplicationContext().getBeanNamesForType(Object.class)); For (String beanName: beanNames) {/ / abstract, filtering (according to the Controller and the RequestMapping annotations in the RequestMappingHandlerMapping filtering) if (! beanName.startsWith(SCOPED_TARGET_NAME_PREFIX) && isHandler(getApplicationContext().getType(beanName))){ DetectHandlerMethods (beanName); } } handlerMethodsInitialized(getHandlerMethods()); } //5. protected void detectHandlerMethods(final Object handler) { Class<? > handlerType = (handler instanceof String ? getApplicationContext().getType((String) handler) : handler.getClass()); final Map<Method, T> mappings = new IdentityHashMap<Method, T>(); final Class<? > userType = ClassUtils.getUserClass(handlerType); / / to get eligible handler Method Set < Method >. The methods = HandlerMethodSelector selectMethods (userType, New MethodFilter() {@override public Boolean matches(Method Method) {RequestMappingInfo = new MethodFilter() {@override public Boolean matches(Method Method) getMappingForMethod(method, userType); if (mapping ! = null) { mappings.put(method, mapping); return true; } else { return false; }}}); // registerHandlerMethod for (Method Method: methods) {registerHandlerMethod(handler, Method, mappings. }}Copy the code
This is how HandlerMapping initializes the mapping
Process summary:
- Get all object subclasses
- Filter the Handle class based on conditions
- Resolves the processing methods defined in the Handle class
- Save the parsed mappings
Take a look at the getHandler(Request) method implementation and see how the DispatcherServlet gets the processing chain
#org.springframework.web.servlet.handler.AbstractHandlerMapping @Override public final HandlerExecutionChain GetHandler (HttpServletRequest Request) throws Exception {// Abstract, call the subclass implementation to get a handler(can be any object, need to be parsed through the HandleAdapter). / / RequestMappingInfoHandlerMapping concrete implementation is match the request path and RequestMapping annotation Object handler = getHandlerInternal (request); . // Wrap handle as HandlerExecutionChain return getHandlerExecutionChain(handler, request); }Copy the code
HandleAdapter
Definition: according to the HandlerMapping. GetHandler () get the Handler of information, the HTTP request parameter parsed and binding
Take a look at the interface definition of the HandlerAdapter
public interface HandlerAdapter {
// Determine whether parsing of this handler type is supported
boolean supports(Object handler);
// The parameters are parsed and handler is called to complete the procedure call
ModelAndView handle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception;
// Last-Modified is used to process HTTP request headers
long getLastModified(HttpServletRequest request, Object handler);
}
Copy the code
RequestMappingHandlerAdapter, for example, first look at the inheritance relationship
You also see that the InitializingBean interface is implemented so let’s start with that
#org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter @Override public void afterPropertiesSet() { // 1. Initialize ControllerAdvice initControllerAdviceCache annotation objects (); / / 2. Loading ArgumentResolver + custom (default) if (this. ArgumentResolvers = = null) {List < HandlerMethodArgumentResolver > resolvers = getDefaultArgumentResolvers(); / / as a Composite object enclosing argumentResolvers = new HandlerMethodArgumentResolverComposite () addResolvers (resolvers); } / / 2. Loading InitBinderArgumentResolvers + custom (default) if (this. InitBinderArgumentResolvers = = null) { List<HandlerMethodArgumentResolver> resolvers = getDefaultInitBinderArgumentResolvers(); / / as a Composite object enclosing initBinderArgumentResolvers = new HandlerMethodArgumentResolverComposite().addResolvers(resolvers); } / / 3. Load ReturnValueHandlers + custom (default) if (this. ReturnValueHandlers = = null) {List < HandlerMethodReturnValueHandler > handlers = getDefaultReturnValueHandlers(); / / as a Composite object enclosing returnValueHandlers = new HandlerMethodReturnValueHandlerComposite () addHandlers (handlers); }} private void initControllerAdviceCache () {/ / from the container to get all the name of the class with ControllerAdvices annotations And packaged into ControllerAdviceBean List<ControllerAdviceBean> beans = ControllerAdviceBean.findAnnotatedBeans(getApplicationContext()); OrderComparator.sort(beans); List<Object> responseBodyAdviceBeans = new ArrayList<Object>(); for (ControllerAdviceBean bean : Beans) {// Select a Method Set<Method> attrMethods = with ModelAttribute and without RequestMapping HandlerMethodSelector.selectMethods(bean.getBeanType(), MODEL_ATTRIBUTE_METHODS); if (! AttrMethods. IsEmpty ()) {/ / save the map in the enclosing modelAttributeAdviceCache. Put (bean, attrMethods); } / / screen with InitBinder annotation Method Set < Method > binderMethods = HandlerMethodSelector. SelectMethods (bean. GetBeanType (), INIT_BINDER_METHODS); if (! BinderMethods. IsEmpty ()) {/ / save the map in the enclosing initBinderAdviceCache. Put (bean, binderMethods); } / / if the class also implements the ResponseBodyAdvice interface added to combine the if (ResponseBodyAdvice. Class. IsAssignableFrom (bean. GetBeanType ())) { responseBodyAdviceBeans.add(bean); }} // Save to the global variable if (! responseBodyAdviceBeans.isEmpty()) { this.responseBodyAdvice.addAll(0, responseBodyAdviceBeans); }}Copy the code
Process summary:
- Loads an object with a ControllerAdvices annotation
- Loading ArgumentResolvers(default + custom)
- Loading InitBinderArgumentResolvers + custom (the default)
- ReturnValueHandlers(default + custom)
Custom extensions will come later
The following is the HandlerAdapter default parser
Look at the HandlerMethodReturnValueHandler interface and the interface HandlerMethodArgumentResolver
// Parameter resolver
public interface HandlerMethodArgumentResolver {
// Determine whether parsing of this parameter is supported (by type, annotation, etc.)
boolean supportsParameter(MethodParameter parameter);
// Parse the parameters to get the result
Object resolveArgument(MethodParameter parameter, ModelAndViewContainer mavContainer, NativeWebRequest webRequest, WebDataBinderFactory binderFactory) throws Exception;
}
// Returns the value parser
public interface HandlerMethodReturnValueHandler {
// Determine whether parsing of the return value is supported (by type, annotation, etc.)
boolean supportsReturnType(MethodParameter returnType);
// Parse the return value
void handleReturnValue(Object returnValue, MethodParameter returnType, ModelAndViewContainer mavContainer, NativeWebRequest webRequest) throws Exception;
}
Copy the code
This is the initialization of the HandlerAdapter
The DispatcherServlet invokes the HandlerAdapter procedure
//1. Call the support() method to check whether handler parsing is supported
HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());
// Call getLastModified() for Get or Head requests to Get the last update time
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 the update time is shorter than the browser cache, it is directly returned to the browser to use the local cache
if (new ServletWebRequest(request, response).checkNotModified(lastModified) && isGet) {
return; }}if(! mappedHandler.applyPreHandle(processedRequest, response)) {return;
}
// Actually invoke the handler.
mv = ha.handle(processedRequest, response, mappedHandler.getHandler());
Copy the code
The DisPatcherServlet calls the HandlerAdapter in three steps:
- Call the support() method to determine whether handler parsing is supported
# org. Springframework. Web. Servlet. DispatcherServlet / / in the doDispatch () method calls the getHandlerAdapter (Object) method to get a HandlerAdapter Protected HandlerAdapter getHandlerAdapter(Object Handler) throws ServletException {// Call the handlerAdapter.support () method For (HandlerAdapter HA: this. HandlerAdapters) {... if (ha.supports(handler)) { return ha; }}... } #org.springframework.web.servlet.mvc.method.AbstractHandlerMethodAdapter @Override public final boolean supports(Object handler) { return (handler instanceof HandlerMethod && supportsInternal((HandlerMethod) handler)); } #org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter @Override protected boolean supportsInternal(HandlerMethod handlerMethod) { return true; }Copy the code
- If it is a Get or Head request, call getLastModified() to Get the last update time
If it is shorter than the browser cache update time, the browser is directly returned to use the local cache
#org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter
@Override
protected long getLastModifiedInternal(HttpServletRequest request, HandlerMethod handlerMethod) {
return -1;
}
Copy the code
- Call the handler() method to complete the procedure call (parameter resolution return value resolution)
#org.springframework.web.servlet.mvc.method.AbstractHandlerMethodAdapter @Override public final ModelAndView handle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { return handleInternal(request, response, (HandlerMethod) handler); } #org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter @Override protected ModelAndView handleInternal(HttpServletRequest request, HttpServletResponse response, Throws Exception {// Handling of HTTP caching headers (expire,cache-control) if (getSessionAttributesHandler(handlerMethod).hasSessionAttributes()) { // Always prevent caching in case of session attribute management. checkAndPrepare(request, response, this.cacheSecondsForSessionAttributeHandlers, true); } else { // Uses configured default cacheSeconds setting. checkAndPrepare(request, response, true); } / / Execute invokeHandlerMethod in a synchronized block, if required, if (this. SynchronizeOnSession) {/ / whether or not to use the session lock HttpSession session = request.getSession(false); if (session ! Object mutex = webutils.getsessionMutex (session); Synchronized (mutex) {return invokeHandleMethod(request, response, handlerMethod); Return invokeHandleMethod(request, response, handlerMethod); } private ModelAndView invokeHandleMethod(HttpServletRequest Request, HttpServletResponse response, HandlerMethod handlerMethod) throws Exception { ServletWebRequest webRequest = new ServletWebRequest(request, response); // handle @initBinder with initBinderAdviceCache WebDataBinderFactory binderFactory = getDataBinderFactory(handlerMethod); / / use modelAttributeAdviceCache to deal with @ ModelAttribute ModelFactory ModelFactory = getModelFactory (handlerMethod, binderFactory); ServletInvocableHandlerMethod requestMappingMethod = createRequestMappingMethod(handlerMethod, binderFactory); ModelAndViewContainer mavContainer = new ModelAndViewContainer(); mavContainer.addAllAttributes(RequestContextUtils.getInputFlashMap(request)); modelFactory.initModel(webRequest, mavContainer, requestMappingMethod); mavContainer.setIgnoreDefaultModelOnRedirect(this.ignoreDefaultModelOnRedirect); // Async processing is not handled by TODO. / / 1 complete procedure call requestMappingMethod. InvokeAndHandle (webRequest, mavContainer); if (asyncManager.isConcurrentHandlingStarted()) { return null; } //2 wrap ModelAndView return getModelAndView(mavContainer, modelFactory, webRequest); }Copy the code
Process Summary of Handle () :
- Execution call requestMappingMethod. InvokeAndHandle (webRequest, mavContainer);
#org.springframework.web.servlet.mvc.method.annotation.ServletInvocableHandlerMethod public void invokeAndHandle(ServletWebRequest webRequest, ModelAndViewContainer mavContainer, Object... ProvidedArgs) throws Exception {// 1.1 Parameter parsing and procedure completion call Object returnValue = invokeForRequest(webRequest, mavContainer, providedArgs); setResponseStatus(webRequest); . Try {/ / 1.2 using returnValueHandlers to deal with returns the result Tell the result to mavContainer process similar parameters in resolving this. ReturnValueHandlers. HandleReturnValue ( returnValue, getReturnValueType(returnValue), mavContainer, webRequest); } } #org.springframework.web.method.support.InvocableHandlerMethod public Object invokeForRequest(NativeWebRequest request, ModelAndViewContainer mavContainer, Object... Object[] args = getMethodArgumentValues(Request, mavContainer, providedArgs); . Object returnValue = doInvoke(args); . return returnValue; } private Object[] getMethodArgumentValues(NativeWebRequest request, ModelAndViewContainer mavContainer, Object... ProvidedArgs) throws Exception {// Parameter information MethodParameter[] parameters = getMethodParameters(); Object[] args = new Object[parameters.length]; for (int i = 0; i < parameters.length; I++) {/ / call HandlerMethodArgumentResolver# supportsParameter judge whether to support the if (this. ArgumentResolvers. SupportsParameter (parameter)) {try {/ / call HandlerMethodArgumentResolver# resolveArgument parsing args[i] = this.argumentResolvers.resolveArgument( parameter, mavContainer, request, this.dataBinderFactory); continue; }... }... } return args; }Copy the code
- Wrapper ModelAndView getModelAndView(mavContainer, modelFactory, webRequest);
// Take the result from mavContainer and wrap it into ModelAndView
private ModelAndView getModelAndView(ModelAndViewContainer mavContainer, ModelFactory modelFactory, NativeWebRequest webRequest) throws Exception {
modelFactory.updateModel(webRequest, mavContainer);
if (mavContainer.isRequestHandled()) {
return null;
}
ModelMap model = mavContainer.getModel();
ModelAndView mav = new ModelAndView(mavContainer.getViewName(), model);
if(! mavContainer.isViewReference()) { mav.setView((View) mavContainer.getView()); }// Redirect request
if (model instanceofRedirectAttributes) { Map<String, ? > flashAttributes = ((RedirectAttributes) model).getFlashAttributes(); HttpServletRequest request = webRequest.getNativeRequest(HttpServletRequest.class); RequestContextUtils.getOutputFlashMap(request).putAll(flashAttributes); }return mav;
}
Copy the code
The call process to this HandlerAdapter is analyzed
SpringMVC mind map