The Servlet with MVC
What is Spring MVC? What is MVC?
Model Data, View View, Controller Controller. Pa! Three things together, MVC comes out.
That simple? Yes, it’s really that simple.
Of course, if you’re not familiar with MVC, read on.
In fact, MVC is a framework pattern for handling Web requests. Let’s think about the flow of user requests:
- Enter the url
- Send the request
- Accept the response
There are only three steps for the user, but there are a lot of steps for the server, and I drew a picture here for you to understand. This is actually missing a lot of what Tomcat itself does for us, and Tomcat is not just Host Context.
Let me explain, the URL that the user sends the request to actually corresponds to a lot of things.
Let’s say localhost, which of course is the IP address. This IP address corresponds to the Host layer in Tomcat.
Context represents a Web application. Remember when you wrote the Servlet project you had a WebApp folder (with web-INF in it and web.xml at the bottom)? It can also be understood that the servlet project was originally written as a Web application, and the user found the application through port mapping of the IP address.
At this point we have found the specified Web application by IP and port, we know that there are multiple servlets in a Web application, and how do we find the corresponding servlet for each request? The answer is still the URL. We go to web.xml to find the Servlet class that has been registered with the application by following /news.
Let me explain the details in accordance with the figure: After the specified Web application is found, the requested path /news is used to search for the corresponding tag in web. XML. The value of the tag’s sub-tag needs to match the requested path. So we get the value of the tag above and then we look to see if any of the tag’s children are equal to the value, and if so, we get the corresponding class at the bottom and parse the request through that class
To summarize, this is the class that finds a matching servlet from the web.xml file using the URL
In fact, this is the native servlet, so where is the shadow of MVC?
Don’t worry, remember that MVC encapsulates servlets, and to understand MVC you need to understand servlets and the relationship between MVC and servlets.
For SpringMVC the DispatcherServlet
Inheritance structure of DispatcherServlet
Did you notice that this DispatcherServlet is actually a Servlet? That said, the core part of Spring MVC is actually a Servlet.
Let me explain the corresponding part briefly.
- FrameworkServlet: Is an abstract parent of DispatcherServlet. It provides the function of loading a corresponding Web application environment, as well as the unified GET, POST, DELETE, PUT and other methods to the DispatcherServlet processing.
- Servlet: A specification that addresses the coupling problem between an HTTP server and business code
- GenericServlet: Raised the scope of the ServletConfig by calling init() with no arguments in the init(ServletConfig) method. Subclasses can override this no-argument initialization method to do some initialization customization (discussed later in the source code analysis).
- HttpServletBean: Servlet configuration information can be automatically assigned to properties of the Servlet as properties of the Bean.
- DispatcherServlet: The last and most important class in the inheritance chain is the core class of the SpringMVC implementation. MVC intercepts all requests by configuring a DispatcherServlet in web.xml, and then dispatches the request through this DispatcherServlet and invokes the appropriate handler to handle the request and response messages.
Do you remember the configuration in web.xml in the SSM framework configuration
<servlet>
<servlet-name>dispatcherServlet</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
<! Send all requests to DispatcherServlet -->
<servlet-mapping>
<servlet-name>dispatcherServlet</servlet-name>
<! -- Intercept all -->
<url-pattern>/</url-pattern>
</servlet-mapping>
Copy the code
Okay, so now that we know that springMVC uses a DispatcherServlet to handle all requests, and we know that it’s definitely not the DispatcherServlet that handles all requests, It’s the @Controller@requestMapping class and the underlying methods that we wrote in the Controller layer. The DispatcherServlet is just a distribution Servlet that dispatches requests to specific processors for us.
So how does this DispatcherServlet work? How does it distribute requests? Just listen to me slowly.
Some of the components that work with DispatcherServlet
I’ll start by simplifying these components and leaving out the unnecessary ones for easier understanding.
To distribute requests and handle requests accordingly, we can be sure that we need to use a Mapping relationship to represent the URL and the corresponding Handler, and a Handler to handle the corresponding request. This brings us to the two most fundamental roles: HandlerMapping and Handler.
So let’s emphasize the work of both.
- HandlerMapping: Establishes the mapping between the request and the handler, that is, we can obtain the corresponding handler through the request.
- Handler: The Handler that processes the request.
So we can draw a simple flow chart again.
Wondering where this HandlerMapping collection comes from? What is the class structure of HandlerMapping? What is the class structure of the Handler?
If so, look ahead with these questions.
First of all, where did this collection of handlerMapping come from? Never mind collections, you don’t even know where the individual came from. So let’s find the answer from the source code. To save you effort, I will directly tell you that the main process of distribution is carried out in the doDispatch method of DispatcherServlet.
Here I present a simplified version of the doDispatch method
public void doDispatcher(HttpServletRequest req, HttpServletResponse resp) throws Exception {
// Use request to get the corresponding handler in HandlerMapping
Object handler = getHandler(req);
if(handler ! =null) {... Call handler's handler}}Copy the code
So what about this getHandler(Request) method? Here I directly put the DispatcherServlet class source
@Nullable
// 这里返回的是 HandlerExecutionChain
// This is a combination of a processor and an interceptor chain
// Return a handler
protected HandlerExecutionChain getHandler(HttpServletRequest request) throws Exception {
if (this.handlerMappings ! =null) {
// Iterate through the handlerMapping and call its getHanlder to get the handler
// Return if not null
for (HandlerMapping mapping : this.handlerMappings) {
HandlerExecutionChain handler = mapping.getHandler(request);
if(handler ! =null) {
returnhandler; }}}return null;
}
Copy the code
We continue to trace the getHandler(Request) method of HandlerMapping.
In fact, if you enter the source code, you will find that HandlerMapping is an interface, so here gives a simple HandlerMapping interface code, if you have the ability to see the source code.
public interface HandlerMapping {
/** * get the request's handler *@paramRequest request *@returnThe processor *@throwsThe Exception Exception * /
Object getHandler(HttpServletRequest request) throws Exception;
}
Copy the code
So what are the concrete implementation classes? Let’s think about it. This mapping is a mapping between the request and the processor. How does it store? What did we do?
Presumably, you already have the answer. When we used the SSM framework, we configured the corresponding annotation (@controller, @reqEustMapping) for the class and method to map the corresponding URL to the processor method.
In idea, you can use CTRL + Alt +B to view method implementation and class implementation inheritance. GetHandler = AbstractHandlerMapping = AbstractHandlerMapping = AbstractHandlerMapping = AbstractHandlerMapping
public final HandlerExecutionChain getHandler(HttpServletRequest request) throws Exception {
// Get handler this is actually an abstract method
// Subclasses can get handler through different implementations
// For example, through the mapping logic of url and name
// Map the url to the corresponding method in requestMapping
Object handler = getHandlerInternal(request);
if (handler == null) {
// Get the default handler if null
// Subclasses can also set their own default handlers
handler = getDefaultHandler();
}
// If not, the DispatcherServlet will return 404
if (handler == null) {
return null;
}
// Bean name or resolved handler?
// If the returned handler is a string, it is considered a beanName
if (handler instanceof String) {
String handlerName = (String) handler;
// Get the appropriate handler from the IOC container via beanName
handler = obtainApplicationContext().getBean(handlerName);
}
// The following is to wrap the processor and interceptor into a processor execution chain
HandlerExecutionChain executionChain = getHandlerExecutionChain(handler, request);
if (logger.isTraceEnabled()) {
logger.trace("Mapped to " + handler);
}
else if(logger.isDebugEnabled() && ! request.getDispatcherType().equals(DispatcherType.ASYNC)) { logger.debug("Mapped to " + executionChain.getHandler());
}
if (CorsUtils.isCorsRequest(request)) {
CorsConfiguration globalConfig = this.corsConfigurationSource.getCorsConfiguration(request); CorsConfiguration handlerConfig = getCorsConfiguration(handler, request); CorsConfiguration config = (globalConfig ! =null ? globalConfig.combine(handlerConfig) : handlerConfig);
executionChain = getCorsHandlerExecutionChain(request, executionChain, config);
}
// Returns the handler execution chain
return executionChain;
}
Copy the code
If you don’t understand the rest, you just need to understand the code in my comment area. The most important thing to take away from this is that the actual handler fetch is in the getHandlerInternal(Request) of the subclass implementation, so let’s look at which subclasses are available.
We can see there are AbstractHandlerMethodMapping, AbstractUrlHandlerMapping, WelcomeHandlerMapping.
We focus on AbstractHandlerMethodMapping (processor) provide method and AbstractUrlHandlerMapping (provide url corresponding processor mapping), here in order not to lose time, We analyze AbstractHandlerMethodMapping directly, it is a method of annotation mapping an abstract class.
@Override
protected HandlerMethod getHandlerInternal(HttpServletRequest request) throws Exception {
// Get the requested path
String lookupPath = getUrlPathHelper().getLookupPathForRequest(request);
this.mappingRegistry.acquireReadLock();
try {
// Get HandlerMethod from lookupPath
// What is the HandlerMethod?
LookupHandlerMethod lookupHandlerMethod lookupHandlerMethod
HandlerMethod handlerMethod = lookupHandlerMethod(lookupPath, request);
return(handlerMethod ! =null ? handlerMethod.createWithResolvedBean() : null);
}
finally {
this.mappingRegistry.releaseReadLock(); }}// The logic here is a little more complicated
// Just know that it matches the return handler method by request
// If there are multiple handler methods that can handle the current Http request, return the best matching handler
@Nullable
protected HandlerMethod lookupHandlerMethod(String lookupPath, HttpServletRequest request) throws Exception {
List<Match> matches = new ArrayList<>();
List<T> directPathMatches = this.mappingRegistry.getMappingsByUrl(lookupPath);
if(directPathMatches ! =null) {
addMatchingMappings(directPathMatches, matches, request);
}
if (matches.isEmpty()) {
// No choice but to go through all mappings...
addMatchingMappings(this.mappingRegistry.getMappings().keySet(), matches, request);
}
if(! matches.isEmpty()) { Comparator<Match> comparator =new MatchComparator(getMappingComparator(request));
matches.sort(comparator);
Match bestMatch = matches.get(0);
if (matches.size() > 1) {
if (logger.isTraceEnabled()) {
logger.trace(matches.size() + " matching mappings: " + matches);
}
if (CorsUtils.isPreFlightRequest(request)) {
return PREFLIGHT_AMBIGUOUS_MATCH;
}
Match secondBestMatch = matches.get(1);
if (comparator.compare(bestMatch, secondBestMatch) == 0) {
Method m1 = bestMatch.handlerMethod.getMethod();
Method m2 = secondBestMatch.handlerMethod.getMethod();
String uri = request.getRequestURI();
throw new IllegalStateException(
"Ambiguous handler methods mapped for '" + uri + "' : {" + m1 + "," + m2 + "}");
}
}
request.setAttribute(BEST_MATCHING_HANDLER_ATTRIBUTE, bestMatch.handlerMethod);
handleMatch(bestMatch.mapping, lookupPath, request);
// Returns the best match
return bestMatch.handlerMethod;
}
else {
return handleNoMatch(this.mappingRegistry.getMappings().keySet(), lookupPath, request); }}Copy the code
Now that the logic is getting more complicated, let’s make a summary: In the DispatcherServlet we get a handler by iterating through the handlerMapping collection and calling its getHandler method. This handler is an Object (because Spring integrates processors from other frameworks, And use these handlers to process the request. HandlerMapping is simply an interface that implements this method for the AbstractHandlerMapping class and provides a custom getHandlerInternal(Request) method for subclasses to get handler. Annotations to identify controller for our general mode method and the url request path mapping is to obtain corresponding HandlerMethod request by AbstractHandlerMethodMapping.
So, again, what is a HandlerMethod?
Remember the question above, where did the HandlerMapping collection come from? What is the class structure of HandlerMapping? What is the class structure of the Handler?
We can now answer the class structure of Handler. Handler is an Object that spring uses in order for third-party framework processors to plug in to handle requests, and for annotation forms a Handler is a HandlerMethod. Here I give a simple implementation of HandlerMethod, if you have the ability to view the source.
@Data
public class HandlerMethod {
// Bean is the object of the class that identifies the Controller annotation
private Object bean;
// The type of the object
privateClass<? > beanType;// Identifies the method annotated by RequestMapping on this class
private Method method;
}
Copy the code
The HandlerMethod holds the controller class and its corresponding methods. Why store them? If you think about it, the @requestMapping annotation identifies the method as a handler. The handler is stored in the HandlerMethod, and all you need to do to call the handler is call the bean’s method via reflection. If you don’t understand this, take a look at the code I wrote below.
// Forget about ModelAndView for now
public ModelAndView handleRequest(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
ModelAndView modelAndView = null;
HandlerMethod handlerMethod = ((HandlerMethod) handler);
// Get the method of HandlerMethod
Method method = handlerMethod.getMethod();
if(method ! =null) {
// Call the method through reflection and return the view object (this is how to handle it)
modelAndView = (ModelAndView)method.invoke(BeanFactory.getBean(handlerMethod.getBeanType()));
}
return modelAndView;
}
Copy the code
Let’s look at the question above. Where does this HandlerMapping collection come from? What is the class structure of HandlerMapping? What is the class structure of the Handler
The third problem is solved, the second problem is solved above, so the first problem comes.
/ / HandlerMapping.getMappings (); / / GetMappings (); / / GetMappings (); / / GetMappings (); / / GetMappings (); / / GetMappings (); / / GetMappings ();
At this point, we have to go back to the roots. Let me show you this picture, if you remember
What can you think of? I’m going to assume that you know something about servlets.
We know that DispatcherServlet is a servlet. A servlet must have an init() method (remember what GenericServlet does above? If you’re not familiar with init(), learn about the servlet life cycle.
It would be a safe guess that the initialization of handlerMappings takes place in the servlet initialization method.
Unfortunately we can’t find init method in DispatcherServlet, so go to his dad, can’t find his grandpa, great grandpa. We know that because DispatcherServlet inherits GenericServlet we need to find the init() no-parameter method to implement. So we found the init() method overridden in the HttpServletBean
@Override
// This guy is not allowed to be overwritten final yet
public final void init(a) throws ServletException {
// Set bean properties from init parameters.
// Store the servlet configuration information into the bean's properties
PropertyValues pvs = new ServletConfigPropertyValues(getServletConfig(), this.requiredProperties);
if(! pvs.isEmpty()) {try {
BeanWrapper bw = PropertyAccessorFactory.forBeanPropertyAccess(this);
ResourceLoader resourceLoader = new ServletContextResourceLoader(getServletContext());
bw.registerCustomEditor(Resource.class, new ResourceEditor(resourceLoader, getEnvironment()));
initBeanWrapper(bw);
bw.setPropertyValues(pvs, true);
}
catch (BeansException ex) {
if (logger.isErrorEnabled()) {
logger.error("Failed to set bean properties on servlet '" + getServletName() + "'", ex);
}
throwex; }}// The important point here is that subclasses are free to play
// This method cannot be overridden because the above step is required
// Remember that the above steps are the responsibility of HttpServletBean
// Continue to read
// Let subclasses do whatever initialization they like.
initServletBean();
}
// Enter the FrameworkServlet to see the implemented initServletBean method
@Override
protected final void initServletBean(a) throws ServletException {
// log
getServletContext().log("Initializing Spring " + getClass().getSimpleName() + "'" + getServletName() + "'");
if (logger.isInfoEnabled()) {
logger.info("Initializing Servlet '" + getServletName() + "'");
}
long startTime = System.currentTimeMillis();
// Here's the thing
try {
// Initialize the container and context
// we need to remember that we are now executing in the FrameworkServlet
/ / we enter initWebApplicationContext method
this.webApplicationContext = initWebApplicationContext();
// The implementation subclass is not given
// Don't worry about it
initFrameworkServlet();
}
// log
catch (ServletException | RuntimeException ex) {
logger.error("Context initialization failed", ex);
throw ex;
}
if (logger.isDebugEnabled()) {
String value = this.enableLoggingRequestDetails ?
"shown which may lead to unsafe logging of potentially sensitive data" :
"masked to prevent unsafe logging of potentially sensitive data";
logger.debug("enableLoggingRequestDetails='" + this.enableLoggingRequestDetails +
"': request parameters and headers will be " + value);
}
if (logger.isInfoEnabled()) {
logger.info("Completed initialization in " + (System.currentTimeMillis() - startTime) + " ms"); }}// Initialize the container and context
protected WebApplicationContext initWebApplicationContext(a) {
// Check whether there is a special root environment
WebApplicationContext rootContext =
WebApplicationContextUtils.getWebApplicationContext(getServletContext());
WebApplicationContext wac = null;
// If there is no dedicated root environment, we usually don't go here
if (this.webApplicationContext ! =null) {
// A context instance was injected at construction time -> use it
wac = this.webApplicationContext;
if (wac instanceof ConfigurableWebApplicationContext) {
ConfigurableWebApplicationContext cwac = (ConfigurableWebApplicationContext) wac;
if(! cwac.isActive()) {// The context has not yet been refreshed -> provide services such as
// setting the parent context, setting the application context id, etc
if (cwac.getParent() == null) {
// The context instance was injected without an explicit parent -> set
// the root application context (if any; may be null) as the parentcwac.setParent(rootContext); } configureAndRefreshWebApplicationContext(cwac); }}}// If it is null
if (wac == null) {
// Check to see if the servlet is registered
// No context instance was injected at construction time -> see if one
// has been registered in the servlet context. If one exists, it is assumed
// that the parent context (if any) has already been set and that the
// user has performed any initialization such as setting the context id
wac = findWebApplicationContext();
}
if (wac == null) {
// Create your own
// No context instance is defined for this servlet -> create a local one
wac = createWebApplicationContext(rootContext);
}
// Determine whether the environment supports refreshing
// If it is supported, it has been refreshed previously
if (!this.refreshEventReceived) {
// Either the context is not a ConfigurableApplicationContext with refresh
// support or the context injected at construction time had already been
// refreshed -> trigger initial onRefresh manually here.
synchronized (this.onRefreshMonitor) {
/ /!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! Focus on
// DispacherServlet is implemented hereonRefresh(wac); }}if (this.publishContext) {
// Publish the context as a servlet context attribute.
String attrName = getServletContextAttributeName();
getServletContext().setAttribute(attrName, wac);
}
return wac;
}
// DispatcherServlet overrides this method
@Override
protected void onRefresh(ApplicationContext context) {
initStrategies(context);
}
// A series of initialization tasks
protected void initStrategies(ApplicationContext context) {
// Don't worry about the previous ones
initMultipartResolver(context);
/ / region
initLocaleResolver(context);
/ / theme
initThemeResolver(context);
// the point is !!!!
// Initialize HandlerMapping
initHandlerMappings(context);
// Initialize the adapter
initHandlerAdapters(context);
// Initialize exception handling
initHandlerExceptionResolvers(context);
initRequestToViewNameTranslator(context);
initViewResolvers(context);
initFlashMapManager(context);
}
Copy the code
Let’s pause for a moment and get our heads together.
The init() method of GenericServlet is overridden in the HttpServletBean to initiate initialization, where the servlet configuration information is assigned to the bean property information. It then calls initServletBean(), which is the custom initialization method for subclasses. FrameworkServlet implements the method and invoke the initWebApplicationContext () method has carried on the containers and context initialization, The onRefresh(ApplicationContext Context) method is called. Here the FrameworkServlet does nothing but subclasses DispatcherServlet in which it calls initStrategies(Context) for initialization.
So let’s move on to initializing the handlerMappings method.
private void initHandlerMappings(ApplicationContext context) {
this.handlerMappings = null;
if (this.detectAllHandlerMappings) {
// Find all HandlerMappings in the ApplicationContext, including ancestor contexts.
// Look for handlerMappings in the application context
Map<String, HandlerMapping> matchingBeans =
BeanFactoryUtils.beansOfTypeIncludingAncestors(context, HandlerMapping.class, true.false);
if(! matchingBeans.isEmpty()) {this.handlerMappings = new ArrayList<>(matchingBeans.values());
// We keep HandlerMappings in sorted order.
AnnotationAwareOrderComparator.sort(this.handlerMappings); }}else {
try {
HandlerMapping hm = context.getBean(HANDLER_MAPPING_BEAN_NAME, HandlerMapping.class);
this.handlerMappings = Collections.singletonList(hm);
}
catch (NoSuchBeanDefinitionException ex) {
// Ignore, we'll add a default HandlerMapping later.}}// Ensure we have at least one HandlerMapping, by registering
// a default HandlerMapping if no other mappings are found.
// We use the default
if (this.handlerMappings == null) {
// Get the default handlerMappings
this.handlerMappings = getDefaultStrategies(context, HandlerMapping.class);
if (logger.isTraceEnabled()) {
logger.trace("No HandlerMappings declared for servlet '" + getServletName() +
"': using default strategies from DispatcherServlet.properties"); }}}protected <T> List<T> getDefaultStrategies(ApplicationContext context, Class<T> strategyInterface) {
String key = strategyInterface.getName();
// Get the contents of defaultStrategies
String value = defaultStrategies.getProperty(key);
if(value ! =null) {
// Parse the corresponding content and initialize handlerMappings
// Get an array of class names in the content
String[] classNames = StringUtils.commaDelimitedListToStringArray(value);
List<T> strategies = new ArrayList<>(classNames.length);
for (String className : classNames) {
try {
// Create it by reflection and add it to the arrayClass<? > clazz = ClassUtils.forName(className, DispatcherServlet.class.getClassLoader()); Object strategy = createDefaultStrategy(context, clazz); strategies.add((T) strategy); }catch (ClassNotFoundException ex) {
throw new BeanInitializationException(
"Could not find DispatcherServlet's default strategy class [" + className +
"] for interface [" + key + "]", ex);
}
catch (LinkageError err) {
throw new BeanInitializationException(
"Unresolvable class definition for DispatcherServlet's default strategy class [" +
className + "] for interface [" + key + "]", err); }}return strategies;
}
else {
return newLinkedList<>(); }}Copy the code
Things are immediately clear, we now know how handlerMapping is enqueued (get the resource content of defaultStrategies traversal content get the class name and create objects via reflection to be enqueued), So it’s safe to assume that there’s a secret in defaultStrategies, which must have defined the class name of the default handlerMapping.
Sure enough, let’s look at the code
private static final Properties defaultStrategies;
// defaultStrategies are already loaded in the static block
static {
// Load default strategy implementations from properties file.
// This is currently strictly internal and not meant to be customized
// by application developers.
try {
// Initialize defaultStrategies with resources
// The resource path here is important !!!!
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()); }}/ / here ah DispatcherServlet. Properties
private static final String DEFAULT_STRATEGIES_PATH = "DispatcherServlet.properties";
Copy the code
We have to find a DispatcherServlet. The properties of the file, the original are defined to us, we can see the default handlerMapping there are two. BeanNameUrlHandlerMapping and RequestMappingHandlerMapping.
Well, now we can finally sum up.
Among the “colleagues” of the simple DispatcherServlet I define are HandlerMapping and Handler. HandlerMapping is loaded when DispatcherServlet is initialized, and then when DispatcherServlet calls the execution method doDispatch(), The handlerMappings collection is iterated to get the corresponding handler. Handler is an Object(because it needs to be adapted to other frameworks’ handlers) and, in annotation mode, a HandlerMethod (which holds an instance of the Controller class and a method that is called using reflection while processing the method). After retrieving the handler, the handler returns a view object and renders the page.
Let’s have another component Adapter
In an “authentic” MVC process, after iterating through handlerMappings and obtaining the corresponding handler, the mappings will not be handled by a HandlerAdapter.
Here I have written a simple adapter interface, the source code is not complex you can directly read the source code
public interface HandlerAdapter {
ModelAndView handleRequest(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception;
boolean support(Object handler);
}
Copy the code
HandleRequest needless to say, the handler that performs the processing, passes in a handler that must eventually call the handler’s execution method. This is a typical adapter pattern.
Support Checks whether the handler is supported by the adapter.
Now that we understand the above process, it’s not hard to add an adapter, but let’s think about why we need to add an adapter, because we know that handler is an Object and its methods are not fixed, If we want to execute processing methods in a DispatcherServlet via a Handler, then we have to do a lot of type judgment, which is very uncomfortable for DispatcherServlet, so we need to extend it through an adapter.
This allows us to write a simple doDispatch method that can be viewed in the source code
public void doDispatcher(HttpServletRequest req, HttpServletResponse resp) throws Exception {
// Use request to get the corresponding handler in HandlerMapping
Object handler = getHandler(req);
if(handler ! =null) {
// Obtain the corresponding adapter from handler
HandlerAdapter handlerAdapter = getHandlerAdapter(handler);
if(handlerAdapter ! =null) {
// Invoke the handler's processing method through the adapter and return the ModelAndView view objectModelAndView modelAndView = handlerAdapter.handleRequest(req, resp, handler); . Process the view and render}}}Copy the code
View resolution
We know that after the doDispatch method calls the HandlerAdapter processing method, it returns a ModelAndView object. What is this ModelAndView object?
Literal, model and view. So MVC’s Model and View. In SpringMVC, ModelAndView is used by the web generator supported by the framework itself. It is a class that connects the background to the web page, which is rarely used nowadays due to the trend of separating the front and back ends, so I’ll just mention it briefly.
We know that after the HandlerAdapter calls the handler it returns a view object, ModelAndView, and after that, The doDispatch method calls processDispatchResult(processedRequest, Response, mappedHandler, mv, dispatchException) to process the view information, This method in turn calls a Render () method to actually render the view.
Very useful RequestResponseBodyMethodProcessor
Remember @requestBody @responseBody @RestController. That’s right, most of us use them now, so how do they work?
The secret comes from the doDispatch() method
mv = ha.handle(processedRequest, response, mappedHandler.getHandler());
Copy the code
This statement begins. This statement calls the handle method of the corresponding adapter and returns the ModelAndView object.
Of course through the previous study, we know the final call to RequestMappingHandlerAdapter handleInternal method of a class.
// Check the source code for this method and you will find that in addition to handling some session issues
// The invokeHandlerMethod is eventually called
@Override
protected ModelAndView handleInternal(HttpServletRequest request, HttpServletResponse response, HandlerMethod handlerMethod) throws Exception {
ModelAndView mav;
checkRequest(request);
// Execute invokeHandlerMethod in synchronized block if required.
// If session synchronization is configured
if (this.synchronizeOnSession) {
// 则获取 session
HttpSession session = request.getSession(false);
if(session ! =null) {
Object mutex = WebUtils.getSessionMutex(session);
synchronized(mutex) { mav = invokeHandlerMethod(request, response, handlerMethod); }}else {
// No HttpSession available -> no mutex necessarymav = invokeHandlerMethod(request, response, handlerMethod); }}else {
// Call the handler method directly if intra-session synchronization is not configured or a session has not been created
// No synchronization on session demanded at all...
mav = invokeHandlerMethod(request, response, handlerMethod);
}
if(! response.containsHeader(HEADER_CACHE_CONTROL)) {if (getSessionAttributesHandler(handlerMethod).hasSessionAttributes()) {
applyCacheSeconds(response, this.cacheSecondsForSessionAttributeHandlers);
}
else{ prepareResponse(response); }}return mav;
}
Copy the code
So let’s look at what’s going on in the invokeHandlerMethod, which looks pretty dense, but we just have to focus on what’s important, what I’m annotating.
protected ModelAndView invokeHandlerMethod(HttpServletRequest request, HttpServletResponse response, HandlerMethod handlerMethod) throws Exception {
Constructing a Web request is essentially a proxy class that encapsulates the request and response
ServletWebRequest webRequest = new ServletWebRequest(request, response);
try {
WebDataBinderFactory binderFactory = getDataBinderFactory(handlerMethod);
ModelFactory modelFactory = getModelFactory(handlerMethod, binderFactory);
// the point is !!!!!
/ / will handlerMethod encapsulated into ServletInvocableHandlerMethod class
ServletInvocableHandlerMethod invocableMethod = createInvocableHandlerMethod(handlerMethod);
if (this.argumentResolvers ! =null) {
invocableMethod.setHandlerMethodArgumentResolvers(this.argumentResolvers);
}
if (this.returnValueHandlers ! =null) {
invocableMethod.setHandlerMethodReturnValueHandlers(this.returnValueHandlers);
}
// Do some configuration for invocableMethod
invocableMethod.setDataBinderFactory(binderFactory);
invocableMethod.setParameterNameDiscoverer(this.parameterNameDiscoverer);
ModelAndViewContainer mavContainer = new ModelAndViewContainer();
mavContainer.addAllAttributes(RequestContextUtils.getInputFlashMap(request));
modelFactory.initModel(webRequest, mavContainer, invocableMethod);
mavContainer.setIgnoreDefaultModelOnRedirect(this.ignoreDefaultModelOnRedirect);
AsyncWebRequest asyncWebRequest = WebAsyncUtils.createAsyncWebRequest(request, response);
asyncWebRequest.setTimeout(this.asyncRequestTimeout);
WebAsyncManager asyncManager = WebAsyncUtils.getAsyncManager(request);
asyncManager.setTaskExecutor(this.taskExecutor);
asyncManager.setAsyncWebRequest(asyncWebRequest);
asyncManager.registerCallableInterceptors(this.callableInterceptors);
asyncManager.registerDeferredResultInterceptors(this.deferredResultInterceptors);
if (asyncManager.hasConcurrentResult()) {
Object result = asyncManager.getConcurrentResult();
mavContainer = (ModelAndViewContainer) asyncManager.getConcurrentResultContext()[0]; asyncManager.clearConcurrentResult(); LogFormatUtils.traceDebug(logger, traceOn -> { String formatted = LogFormatUtils.formatValue(result, ! traceOn);return "Resume with async result [" + formatted + "]";
});
invocableMethod = invocableMethod.wrapConcurrentResult(result);
}
// the point is !!!!! Call ServletInvocableHandlerMethod invokeAndHandle method
invocableMethod.invokeAndHandle(webRequest, mavContainer);
if (asyncManager.isConcurrentHandlingStarted()) {
return null;
}
return getModelAndView(mavContainer, modelFactory, webRequest);
}
finally{ webRequest.requestCompleted(); }}Copy the code
To summarize the above method is: will encapsulate HandlerMethod object into ServletInvocableHandlerMethod then do some configuration and call it invokeAndHandle method
public void invokeAndHandle(ServletWebRequest webRequest, ModelAndViewContainer mavContainer, Object... providedArgs) throws Exception {
// This step is important to execute the request and get the return value
// The RequestBody annotation is involved here
Object returnValue = invokeForRequest(webRequest, mavContainer, providedArgs);
setResponseStatus(webRequest);
// Process the return value
if (returnValue == null) {
if(isRequestNotModified(webRequest) || getResponseStatus() ! =null || mavContainer.isRequestHandled()) {
disableContentCachingIfNecessary(webRequest);
mavContainer.setRequestHandled(true);
return; }}else if (StringUtils.hasText(getResponseStatusReason())) {
mavContainer.setRequestHandled(true);
return;
}
mavContainer.setRequestHandled(false);
Assert.state(this.returnValueHandlers ! =null."No return value handlers");
try {
this.returnValueHandlers.handleReturnValue(
returnValue, getReturnValueType(returnValue), mavContainer, webRequest);
}
catch (Exception ex) {
if (logger.isTraceEnabled()) {
logger.trace(formatErrorForReturnValue(returnValue), ex);
}
throwex; }}Copy the code
Let’s first parse the Object returnValue = invokeForRequest(webRequest, mavContainer, providedArgs) method, which involves the use of the @RequestBody annotation.
@Nullable
public Object invokeForRequest(NativeWebRequest request, @Nullable ModelAndViewContainer mavContainer, Object... providedArgs) throws Exception {
There are a lot of things you can do with parameters and requests
// This is what we need to look into the source code
Object[] args = getMethodArgumentValues(request, mavContainer, providedArgs);
if (logger.isTraceEnabled()) {
logger.trace("Arguments: " + Arrays.toString(args));
}
// Return the result of the call. It is very simple to call the method through reflection
return doInvoke(args);
}
protected Object[] getMethodArgumentValues(NativeWebRequest request, @Nullable ModelAndViewContainer mavContainer,
Object... providedArgs) throws Exception {
// It's easy to get parameters
MethodParameter[] parameters = getMethodParameters();
if (ObjectUtils.isEmpty(parameters)) {
return EMPTY_ARGS;
}
Object[] args = new Object[parameters.length];
// Iterate over the parameters
for (int i = 0; i < parameters.length; i++) {
MethodParameter parameter = parameters[i];
parameter.initParameterNameDiscovery(this.parameterNameDiscoverer);
args[i] = findProvidedArgument(parameter, providedArgs);
if(args[i] ! =null) {
continue;
}
if (!this.resolvers.supportsParameter(parameter)) {
throw new IllegalStateException(formatArgumentError(parameter, "No suitable resolver"));
}
try {
// the point is !!!!
// Parses the information in the request through the parameter parser to the corresponding parameter
// Return the array of parameters after the final iteration
args[i] = this.resolvers.resolveArgument(parameter, mavContainer, request, this.dataBinderFactory);
}
catch (Exception ex) {
// Leave stack trace for later, exception may actually be resolved and handled...
if (logger.isDebugEnabled()) {
String exMsg = ex.getMessage();
if(exMsg ! =null&&! exMsg.contains(parameter.getExecutable().toGenericString())) { logger.debug(formatArgumentError(parameter, exMsg)); }}throwex; }}return args;
}
@Override
@Nullable
public Object resolveArgument(MethodParameter parameter, @Nullable ModelAndViewContainer mavContainer, NativeWebRequest webRequest, @Nullable WebDataBinderFactory binderFactory) throws Exception {
// Get the corresponding parameter parser
HandlerMethodArgumentResolver resolver = getArgumentResolver(parameter);
if (resolver == null) {
throw new IllegalArgumentException("Unsupported parameter type [" +
parameter.getParameterType().getName() + "]. supportsParameter should be called first.");
}
// Parsing parameters through the parser is important because of the @requestBody annotation
/ / we will call to RequestResponseBodyProcessor class of this method
return resolver.resolveArgument(parameter, mavContainer, webRequest, binderFactory);
}
@Override
public Object resolveArgument(MethodParameter parameter, @Nullable ModelAndViewContainer mavContainer, NativeWebRequest webRequest, @Nullable WebDataBinderFactory binderFactory) throws Exception {
parameter = parameter.nestedIfOptional();
MessageConverters MessageConverters MessageConverters MessageConverters @requestBody
// Due to limited space here will not be in-depth analysis if you want to find the answer to view the source code along the line
Object arg = readWithMessageConverters(webRequest, parameter, parameter.getNestedGenericParameterType());
String name = Conventions.getVariableNameForParameter(parameter);
if(binderFactory ! =null) {
WebDataBinder binder = binderFactory.createBinder(webRequest, arg, name);
if(arg ! =null) {
validateIfApplicable(binder, parameter);
if (binder.getBindingResult().hasErrors() && isBindExceptionRequired(binder, parameter)) {
throw newMethodArgumentNotValidException(parameter, binder.getBindingResult()); }}if(mavContainer ! =null) { mavContainer.addAttribute(BindingResult.MODEL_KEY_PREFIX + name, binder.getBindingResult()); }}return adaptArgumentIfNecessary(arg, parameter);
}
Copy the code
I also put the readWithMessageConverters method of code below, in fact it is mainly through news converter, and then by the transformation of the converter to perform HTTP message to parameters.
for(HttpMessageConverter<? > converter :this.messageConverters) { Class<HttpMessageConverter<? >> converterType = (Class<HttpMessageConverter<? >>) converter.getClass(); GenericHttpMessageConverter<? > genericConverter = (converterinstanceofGenericHttpMessageConverter ? (GenericHttpMessageConverter<? >) converter :null);
if(genericConverter ! =null? genericConverter.canRead(targetType, contextClass, contentType) : (targetClass ! =null && converter.canRead(targetClass, contentType))) {
if(message.hasBody()) { HttpInputMessage msgToUse = getAdvice().beforeBodyRead(message, parameter, targetType, converterType); body = (genericConverter ! =null ? genericConverter.read(targetType, contextClass, msgToUse) :
((HttpMessageConverter<T>) converter).read(targetClass, msgToUse));
body = getAdvice().afterBodyRead(body, msgToUse, parameter, targetType, converterType);
}
else {
body = getAdvice().handleEmptyBody(null, message, parameter, targetType, converterType);
}
break; }}Copy the code
Now that you know how the @requestBody annotation works, the @responseBody annotation immediately comes to the surface. The answer lies in ServletInvocableHandlerMethod obtained after the returnValue invokeAndHandle approach in class
// The answer is right here
try {
this.returnValueHandlers.handleReturnValue(
returnValue, getReturnValueType(returnValue), mavContainer, webRequest);
}
@Override
public void handleReturnValue(@Nullable Object returnValue, MethodParameter returnType, ModelAndViewContainer mavContainer, NativeWebRequest webRequest) throws Exception {
HandlerMethodReturnValueHandler handler = selectHandler(returnValue, returnType);
if (handler == null) {
throw new IllegalArgumentException("Unknown return value type: " + returnType.getParameterType().getName());
}
handler.handleReturnValue(returnValue, returnType, mavContainer, webRequest);
}
Copy the code
The above two methods you can track the source code, actually finally call or in RequestResponseBodyMethodProcessor this class.
@Override
public void handleReturnValue(@Nullable Object returnValue, MethodParameter returnType, ModelAndViewContainer mavContainer, NativeWebRequest webRequest)
throws IOException, HttpMediaTypeNotAcceptableException, HttpMessageNotWritableException {
mavContainer.setRequestHandled(true);
// Encapsulate the Web request
ServletServerHttpRequest inputMessage = createInputMessage(webRequest);
ServletServerHttpResponse outputMessage = createOutputMessage(webRequest);
// The returned value is parsed through the message parser
// Try even with null return value. ResponseBodyAdvice could get involved.
writeWithMessageConverters(returnValue, returnType, inputMessage, outputMessage);
}
/ / here I posted writeWithMessageConverters method of main code Because this method is a little long.
// The Http message converter collection is iterated here
for(HttpMessageConverter<? > converter :this.messageConverters) {
GenericHttpMessageConverter genericConverter = (converter instanceofGenericHttpMessageConverter ? (GenericHttpMessageConverter<? >) converter :null);
if(genericConverter ! =null? ((GenericHttpMessageConverter) converter).canWrite(targetType, valueType, selectedMediaType) : converter.canWrite(valueType, selectedMediaType)) { body = getAdvice().beforeBodyWrite(body, returnType, selectedMediaType, (Class<? extends HttpMessageConverter<? >>) converter.getClass(), inputMessage, outputMessage);if(body ! =null) {
Object theBody = body;
LogFormatUtils.traceDebug(logger, traceOn ->
"Writing ["+ LogFormatUtils.formatValue(theBody, ! traceOn) +"]");
addContentDispositionHeader(inputMessage, outputMessage);
// Output key...
if(genericConverter ! =null) {
genericConverter.write(body, targetType, selectedMediaType, outputMessage);
}
else{ ((HttpMessageConverter) converter).write(body, selectedMediaType, outputMessage); }}else {
if (logger.isDebugEnabled()) {
logger.debug("Nothing to write: null body"); }}return; }}Copy the code
To summarize: we at the time of call HandlerAdapter processing method will jump call to RequestMappingHandlerAdapter handleInternal method. There will be the original processor HandlerMethod encapsulated into ServletInvocableHandlerMethod, then call invokeAndHandle methods in this class, this method mainly has carried on the corresponding method processor method calls, Before the call, the content in the Http packet is converted to the corresponding parameter content. After the call completes and returns the returnValue, the corresponding HttpMessageConvert conversion method is called and returned.
What did it end up like?
Now, do you understand Spring MVC?