The difference between traditional tomcat startup service and springboot startup built-in tomcat service (recommended)

The difference between traditional tomcat startup service and springboot startup built-in tomcat service (recommended)

Spring integration with springmvc

  • The web.xml configuration of spring integration springmvc is as follows. Tomcat will load the content in web.xml during the startup process. ContextLoaderListener implements the ServletContextListener interface in tomcat. Therefore, during the startup process of the tomcat container, the spring container is initialized through ContextLoaderListener, and the spring configuration file specified by classpath:spring/applicationContext-*.xml is loaded. In this configuration file, I only configured <context:component-scan base-package="org.com.yp"/>, which means scanning the classes under the org.com.yp package, including @Component @Controller@Service and other annotations, for bean registration.
  • Bean registration is done by loading bean definitions through the AbstractXmlApplicationContext.loadBeanDefinitions method of this class.

The bean definition loaded in spring is loaded by the ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory() method in the org.springframework.context.ConfigurableApplicationContext#refresh method. This method then calls the org.springframework.context.support.AbstractRefreshableApplicationContext#refreshBeanFactory method to create a bean factory and load the bean definition.

<?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>Archetype Created Web Application</display-name>
  <listener>
    <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
  </listener>
  <!-- Load the spring container-->
  <context-param>
    <param-name>contextConfigLocation</param-name>
    <param-value>classpath:spring/applicationContext-*.xml</param-value>
  </context-param>
​
  <servlet>
    <servlet-name>mvc-dispatcher</servlet-name>
    <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
    <!-- Configure the configuration file that springMVC needs to load-->
    <init-param>
      <param-name>contextConfigLocation</param-name>
      <param-value>classpath:spring/spring-*.xml</param-value>
    </init-param>
  </servlet>
​
  <servlet-mapping>
    <servlet-name>mvc-dispatcher</servlet-name>
    <!-- Matches all requests by default -->
    <url-pattern>/</url-pattern>
  </servlet-mapping>
</web-app>

When the tomcat container is started, when accessing resources through the path, the org.springframework.web.servlet.HttpServletBean#init method will be called for the first time, and subsequent http requests will no longer use this method class; HttpServletBean implements the specification of the Servlet interface, so when the browser request is initialized and executed through the servlet interface, the loading class defined in the springmvc configuration will be loaded from the spring container. Spring and springmvc are parent-child containers. The following is the init method of HttpServletBean

public final void init() throws ServletException {
		// Set bean properties from init parameters.
		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);
				}
				throw ex;
			}
		}

        //Finally, the refresh method of the org.springframework.context.ConfigurableApplicationContext#refresh container will be called.
        // Initialize the springmvc container initServletBean();
	}
    }

Springboot starts the container

  • The way springboot is started is to first initialize the spring context environment (including the bean factory) in the org.springframework.boot.SpringApplication#run(java.lang.String...) method of springboot, and then call the ConfigurableApplicationContext#refresh method in the Spring container to initialize the bean through the org.springframework.boot.SpringApplication#refreshContext method.
  • In the environment of spring and springmvc integration, bean definitions are loaded in the org.springframework.context.support.AbstractApplicationContext#obtainFreshBeanFactory method, while in springboot it is in

org.springframework.context.support.AbstractApplicationContext#invokeBeanFactoryPostProcessors method, in which the bean definition is loaded through the ConfigurationClassPostProcessor class, which implements the BeanDefinitionRegistryPostProcessor interface, which allows the processing of bean definitions.

// The BeanDefinitionRegistryPostProcessor in spring is a subinterface of BeanFactoryPostProcessor.
// The function of BeanFactoryPostProcessor is to execute the postProcessBeanFactory() method when the bean definition information has been loaded but not initialized.
// The BeanDefinitionRegistryPostProcessor is executed before the BeanFactoryPostProcessor. The execution order is defined in the source code // org.springframework.context.support.PostProcessorRegistrationDelegate#invokeBeanFactoryPostProcessors() method // BeanFactoryPostProcessor is the bean property processing container of the bean factory. In simple terms, it can manage all beandefinition (uninstantiated) data in our bean factory and can modify properties at will.
public void refresh() throws BeansException, IllegalStateException {
        synchronized (this.startupShutdownMonitor) {
            prepareRefresh();
            
            //Get and tell the subclass to initialize the Bean factory to load the beans into the cache. Spring Spring MVC integration is to initialize the bean here. ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();
            
            prepareBeanFactory(beanFactory);
​
            try {
                postProcessBeanFactory(beanFactory);
​
                // The springboot container starts loading at this point and initializes the following bean names
                //0 = "org.springframework.context.annotation.internalConfigurationAnnotationProcessor" =》Corresponding ConfigurationClassPostProcessor class //1 = "org.springframework.context.annotation.internalAutowiredAnnotationProcessor" =》AutowiredAnnotationBeanPostProcessor
                //2 = "org.springframework.context.annotation.internalCommonAnnotationProcessor" =》CommonAnnotationBeanPostProcessor
                //3 = "org.springframework.context.event.internalEventListenerProcessor" =》EventListenerMethodProcessor
                //4 = "org.springframework.context.event.internalEventListenerFactory" = "DefaultEventListenerFactory
                // Call the postprocessor of our bean factory. Load the bean definition (not instantiation), load the scan path in the startup class through ConfigurationClassPostProcessor // Then load the path to the bean invokeBeanFactoryPostProcessors(beanFactory);
​
                registerBeanPostProcessors(beanFactory);
​
                initMessageSource();
​
                initApplicationEventMulticaster();
​
                // This method is also implemented by a subclass of Springboot, which also starts tomat from this method.
                onRefresh();
​
                registerListeners();
​
                //Instantiate the rest of our singleton beans.
                finishBeanFactoryInitialization(beanFactory);
​
                // Finally, the container refreshes and publishes the refresh event (Spring cloud is also started from here)
                finishRefresh();
            }
​
            catch (BeansException ex) {
                if (logger.isWarnEnabled()) {
                    logger.warn("Exception encountered during context initialization - " +
                            "cancelling refresh attempt: " + ex);
                }
​
                // Destroy already created singletons to avoid dangling resources.
                destroyBeans();
​
                // Reset 'active' flag.
                cancelRefresh(ex);
​
                // Propagate exception to caller.
                throw ex;
            }
​
            finally {
                // Reset common introspection caches in Spring's core, since we
                // might not ever need metadata for singleton beans anymore...
                resetCommonCaches();
            }
        }
    }

This is the end of this article about the difference between traditional tomcat startup service and springboot startup built-in tomcat service. For more information about the difference between tomcat startup service and springboot startup built-in tomcat service, please search 123WORDPRESS.COM's previous articles or continue to browse the following related articles. I hope you will support 123WORDPRESS.COM in the future!

You may also be interested in:
  • Detailed explanation of the startup principle of SpringBoot built-in tomcat
  • How to cancel the built-in Tomcat startup in SpringBoot and use external Tomcat instead
  • A brief discussion on the difference between springboot built-in tomcat and external independent deployment of tomcat
  • Tomcat reports an error when starting the springboot project war package: Error when starting the child
  • How to start the Tomcat process in SpringBoot
  • Source code analysis of the process of starting the built-in Tomcat in SpringBoot application

<<:  Is a design that complies with design specifications a good design?

>>:  Implement MaterialUI button click animation based on CSS and encapsulate it into a React component

Recommend

HTML+CSS+JavaScript to create a simple tic-tac-toe game

Table of contents Implementing HTML Add CSS Imple...

Tips on MySQL query cache

Table of contents Preface Introduction to QueryCa...

MYSQL slow query and log settings and testing

1. Introduction By enabling the slow query log, M...

Explaining immutable values ​​in React

Table of contents What are immutable values? Why ...

How to quickly log in to MySQL database without password under Shell

background When we want to log in to the MySQL da...

MySQL UNION operator basic knowledge points

MySQL UNION Operator This tutorial introduces the...

Delete the image operation of none in docker images

Since I usually use the docker build command to g...

Summary of methods for writing judgment statements in MySQL

How to write judgment statements in mysql: Method...

Implementation of Nginx+ModSecurity security module deployment

Table of contents 1. Download 2. Deployment 1.Ngi...

Dynamic starry sky background implemented with CSS3

Result:Implementation Code html <link href=...

CSS Back to Top Code Example

Most websites nowadays have long pages, some are ...

Detailed tutorial on installing pxc cluster with docker

Table of contents Preface Preliminary preparation...

About React Native unable to link to the simulator

React Native can develop iOS and Android native a...