平时开发springboot项目的时候,一个SpringApplication注解加一个main方法就能够启动服务器运行起来(默认tomcat),看了下源码,这里讲下认为主要的流程java
主要流程以下web
0.启动main方法开始spring
1.初始化配置:经过类加载器,(loadFactories)读取classpath下全部的spring.factories配置文件,建立一些初始配置对象;通知监听者应用程序启动开始,建立环境对象environment,用于读取环境配置 如 application.yml数组
2.建立应用程序上下文-createApplicationContext,建立 bean工厂对象缓存
3.刷新上下文(启动核心)
3.1 配置工厂对象,包括上下文类加载器,对象发布处理器,beanFactoryPostProcessor
3.2 注册并实例化bean工厂发布处理器,而且调用这些处理器,对包扫描解析(主要是class文件)
3.3 注册并实例化bean发布处理器 beanPostProcessor
3.4 初始化一些与上下文有特别关系的bean对象(此处启动tomcat服务器)
3.5 实例化全部bean工厂缓存的bean对象(剩下的)
3.6 发布通知-通知上下文刷新完成tomcat
4.通知监听者-启动程序完成springboot
启动中,大部分对象都是BeanFactory对象经过反射建立服务器
SpringBoot的启动解析代码过多,下文是总体流程的部分主要代码app
启动程序:jvm
import org.springframework.boot.SpringApplication;//启动类 import org.springframework.boot.autoconfigure.SpringBootApplication; @SpringBootApplication //启动必要注解 public class YourApplication { //运行main方法启动springboot public static void main(String[] args) { SpringApplication.run(YourApplication.class, args);//启动类静态run方法 } }
org.springframework.boot.SpringApplication
包含主流程方法
启动类在运行静态run方法的时候,是先建立一个SpringApplication对象,再运行对象的run方法,工厂初始配置在构造函数中完成,run方法定义运行整体流程
// 静态方法 org.springframework.boot.SpringApplication.run(Class<?>[], String[]) public static ConfigurableApplicationContext run(Class<?>[] primarySources, String[] args) { return new SpringApplication(primarySources).run(args); } // 构造方法 public SpringApplication(ResourceLoader resourceLoader, Class<?>... primarySources) { //.......... //// 1.(loadFactories)读取classpath下全部的spring.factories配置文件 //// // 配置应用程序启动前的初始化对象 setInitializers((Collection) getSpringFactoriesInstances(ApplicationContextInitializer.class)); // 配置应用程序启动前的监听器 setListeners((Collection) getSpringFactoriesInstances(ApplicationListener.class)); this.mainApplicationClass = deduceMainApplicationClass(); } // 对象run方法 开始启动程序 public ConfigurableApplicationContext run(String... args) { //...... // 通知监听者启动开始 listeners.starting(); try { // 建立应用程序环境 配置文件在此处读取(application.properties application.yml) ConfigurableEnvironment environment = prepareEnvironment(listeners, applicationArguments); //// 2.建立应用程序上下文...此处建立了beanfactory //// context = createApplicationContext(); //// 3.刷新上下文(spring启动核心) //// refreshContext(context); //// 4.启动完成通知...... //// listeners.started(context); } catch (Throwable ex) { handleRunFailure(context, ex, exceptionReporters, listeners); throw new IllegalStateException(ex); } try { listeners.running(context); } catch (Throwable ex) { handleRunFailure(context, ex, exceptionReporters, null); throw new IllegalStateException(ex); } return context; }
springboot启动应用程序以前,会建立一些初始化对象和监听器
这个操做在构造方法中完成,根据配置文件,建立ApplicationContextInitializer.class
,ApplicationListener.class
两个接口的实现类,至于具体建立那些类对象,根据下面的方法逻辑去作
org.springframework.boot.SpringApplication.getSpringFactoriesInstances()
->
org.springframework.core.io.support.SpringFactoriesLoader.loadFactoryNames()
->
org.springframework.core.io.support.SpringFactoriesLoader.loadSpringFactories()
->
createSpringFactoriesInstances()
//构造方法中的初始化对象建立 setInitializers((Collection) getSpringFactoriesInstances(ApplicationContextInitializer.class)); setListeners((Collection) getSpringFactoriesInstances(ApplicationListener.class)); //看一下getSpringFactoriesInstances方法 private <T> Collection<T> getSpringFactoriesInstances(Class<T> type) { return getSpringFactoriesInstances(type, new Class<?>[] {}); } private <T> Collection<T> getSpringFactoriesInstances(Class<T> type, Class<?>[] parameterTypes, Object... args) { ClassLoader classLoader = getClassLoader(); // 获取初始化类的类名 Set<String> names = new LinkedHashSet<>(SpringFactoriesLoader.loadFactoryNames(type, classLoader)); // 经过这些类名实例化对象 List<T> instances = createSpringFactoriesInstances(type, parameterTypes, classLoader, args, names); AnnotationAwareOrderComparator.sort(instances); return instances; } // 读取配置方法 // 更详深层的代码在org.springframework.core.io.support.SpringFactoriesLoader.loadSpringFactories(ClassLoader) public static List<String> loadFactoryNames(Class<?> factoryType, @Nullable ClassLoader classLoader) { String factoryTypeName = factoryType.getName(); return loadSpringFactories(classLoader).getOrDefault(factoryTypeName, Collections.emptyList()); } // loadSpringFactories(classLoader)读取运行环境中全部META-INF/spring.factories配置
经过上面的方法,
spring-boot-2.2.8.RELEASE.jar/META-INF/spring.factories的文件中是这样,
# Application Context Initializers org.springframework.context.ApplicationContextInitializer=\ org.springframework.boot.context.ConfigurationWarningsApplicationContextInitializer,\ org.springframework.boot.context.ContextIdApplicationContextInitializer,\ org.springframework.boot.context.config.DelegatingApplicationContextInitializer,\ org.springframework.boot.rsocket.context.RSocketPortInfoApplicationContextInitializer,\ org.springframework.boot.web.context.ServerPortInfoApplicationContextInitializer
若是只读取这一个文件,loadFactoryNames(ApplicationContextInitializer.class,classLoader)
读取返回的就是下面的数组:
[org.springframework.context.ApplicationContextInitializer, org.springframework.boot.context.ConfigurationWarningsApplicationContextInitializer, org.springframework.boot.context.ContextIdApplicationContextInitializer, org.springframework.boot.context.config.DelegatingApplicationContextInitializer, org.springframework.boot.web.context.ServerPortInfoApplicationContextInitializer]
经过 Class.forName(className)
获取这些类的Class,最后反射newInstance
建立这些对象
建立好这些对象后,启动监听器
listeners.starting(); // 这里也是一些调用操做
读取application配置
监听器启动以后,会读取application.properties 或者 application.yml文件
ConfigurableEnvironment environment = prepareEnvironment(listeners, applicationArguments); //此处application.properties配置文件会被读取
建立应用上下文
初始化和配置好后,开始建立应用程序上下文,createApplicationContext ,关键的工厂BeanFactory就是此处建立,具体逻辑以下
// 建立应用程序上下文 context = createApplicationContext(); protected ConfigurableApplicationContext createApplicationContext() { // 上下文建立的判断逻辑 Class<?> contextClass = this.applicationContextClass; if (contextClass == null) { try { switch (this.webApplicationType) { case SERVLET: contextClass = Class.forName(DEFAULT_SERVLET_WEB_CONTEXT_CLASS); break; case REACTIVE: contextClass = Class.forName(DEFAULT_REACTIVE_WEB_CONTEXT_CLASS); break; default: contextClass = Class.forName(DEFAULT_CONTEXT_CLASS); } } catch (ClassNotFoundException ex) { throw new IllegalStateException( "Unable create a default ApplicationContext, please specify an ApplicationContextClass", ex); } } return (ConfigurableApplicationContext) BeanUtils.instantiateClass(contextClass); } public static final String DEFAULT_SERVLET_WEB_CONTEXT_CLASS = "org.springframework.boot." + "web.servlet.context.AnnotationConfigServletWebServerApplicationContext"; // 默认是建立这个类
这里经过this.webApplicationType判断建立具体的应用上下文,也是反射建立对象,默认建立的是org.springframework.boot.web.servlet.context.AnnotationConfigServletWebServerApplicationContext
对象,看一下这个类的基本信息
public class AnnotationConfigServletWebServerApplicationContext extends ServletWebServerApplicationContext implements AnnotationConfigRegistry { // 构造方法 public AnnotationConfigServletWebServerApplicationContext() { this.reader = new AnnotatedBeanDefinitionReader(this); this.scanner = new ClassPathBeanDefinitionScanner(this); } }
建立工厂对象
此类继承了不少类,其中一个父类是org.springframework.context.support.GenericApplicationContext
jvm机制,建立对象的时候,先运行父类的构造方法,因此建立了beanFactory
// 超级父类 GenericApplicationContext的构造方法 public GenericApplicationContext() { this.beanFactory = new DefaultListableBeanFactory();//建立工厂对象 }
建立好上下文以后,开始刷新上下文,这里作了不少
工厂配置,bean处理器配置,类的扫描,解析,bean定义,bean类信息缓存,服务器建立,bean实例化,动态代理对象的建立等,
spring中注册bean信息和实例化bean是两件事情。
注册bean信息不是建立bean对象,是解析bean类获取详细信息,会建立BeanDefinition对象,携带bean类的字节码和方法等信息,把BeanDefinition对象注册保存到工厂BeanDefinitionMap中。
工厂实例化bean时直接BeanDefinitionMap.get(beanName) 获取bean的字节码信息,经过反射建立对象,而后将bean对象保存到singletonObjects中。
refreshContext(context); //刷新上下文
默认实际对应的是org.springframework.context.support.AbstractApplicationContext
类的refresh()
方法
@Override public void refresh() throws BeansException, IllegalStateException { synchronized (this.startupShutdownMonitor) { //...... // 3.1配置工厂对象 prepareBeanFactory(beanFactory); try { postProcessBeanFactory(beanFactory); // 3.2注册并实例化bean工厂处理器,并调用他们 invokeBeanFactoryPostProcessors(beanFactory); // 3.3注册并实例化bean处理器 registerBeanPostProcessors(beanFactory); // 3.4 初始化一些与上下文有特别关系的bean对象(此处启动tomcat服务器) onRefresh(); // 3.5 实例化全部bean工厂缓存的bean对象(剩下的). finishBeanFactoryInitialization(beanFactory); // 3.6 发布通知-通知上下文刷新完成 finishRefresh(); } catch (BeansException ex) {// ......Propagate exception to caller. throw ex; } finally {// ...... resetCommonCaches(); } } }
工厂建立好后,首先配置的是类加载器,而后是一些对象发布处理器(拦截器)
//// 3.1配置工厂对象 prepareBeanFactory(beanFactory); protected void prepareBeanFactory(ConfigurableListableBeanFactory beanFactory) { // 设置类加载器 beanFactory.setBeanClassLoader(getClassLoader()); beanFactory.setBeanExpressionResolver(new StandardBeanExpressionResolver(beanFactory.getBeanClassLoader())); beanFactory.addPropertyEditorRegistrar(new ResourceEditorRegistrar(this, getEnvironment())); // 添加BeanPostProcessor beanFactory.addBeanPostProcessor(new ApplicationContextAwareProcessor(this)); beanFactory.ignoreDependencyInterface(EnvironmentAware.class); beanFactory.ignoreDependencyInterface(EmbeddedValueResolverAware.class); // ...... } }
过程主要是工厂发布处理器的建立和调用,逻辑较多
//// 3.2注册并实例化bean工厂处理器,并调用他们 invokeBeanFactoryPostProcessors(beanFactory); protected void invokeBeanFactoryPostProcessors(ConfigurableListableBeanFactory beanFactory) { PostProcessorRegistrationDelegate.invokeBeanFactoryPostProcessors(beanFactory, getBeanFactoryPostProcessors()); // ...... } // PostProcessorRegistrationDelegate.invokeBeanFactoryPostProcessors public static void invokeBeanFactoryPostProcessors( ConfigurableListableBeanFactory beanFactory, List<BeanFactoryPostProcessor> beanFactoryPostProcessors) { postProcessorNames = beanFactory.getBeanNamesForType(BeanDefinitionRegistryPostProcessor.class, true, false); for (String ppName : postProcessorNames) { if (!processedBeans.contains(ppName) && beanFactory.isTypeMatch(ppName, Ordered.class)) { // 建立BeanDefinitionRegistryPostProcessor处理器 currentRegistryProcessors.add(beanFactory.getBean(ppName, BeanDefinitionRegistryPostProcessor.class)); processedBeans.add(ppName); } } // 调用这些处理器 invokeBeanDefinitionRegistryPostProcessors(currentRegistryProcessors, registry); // ... } // 循环调用 private static void invokeBeanDefinitionRegistryPostProcessors( Collection<? extends BeanDefinitionRegistryPostProcessor> postProcessors, BeanDefinitionRegistry registry) { for (BeanDefinitionRegistryPostProcessor postProcessor : postProcessors) { postProcessor.postProcessBeanDefinitionRegistry(registry); } }
BeanDefinitionRegistryPostProcessor的子类对象在此处建立并调postProcessBeanDefinitionRegistry
方法。
其中org.springframework.context.annotation.ConfigurationClassPostProcessor
就是BeanDefinitionRegistryPostProcessor的子类,是一个spring的类解析器,扫描包下全部的类,解析出bean类,注册到bean工厂由此类主要参与,其中有很多递归
//// 3.3注册并实例化bean处理器 registerBeanPostProcessors(beanFactory);
BeanFactoryPostProcessors 和 BeanPostProcessors是有区别的
BeanFactoryPostProcessors 是工厂发布处理器,定义什么是bean,知道哪些是bean类,解析class文件,包括注解解析,成员对象依赖解析等;BeanPostProcessors主要在BeanFactoryPostProcessors调用完以后工做
通常在bean对像的建立以前或以后,BeanFactory调用这些bean处理器拦截处理,Spring代理对象的建立也是经过beanPostProcessor处理器来实现
AnnotationAwareAspectJAutoProxyCreator实现了BeanPostProcessors,在bean被工厂建立以后,BeanFactory调用拦截器的postProcessAfterInitialization作拦截处理。此拦截器处理器实际执行的是父类org.springframework.aop.framework.autoproxy.AbstractAutoProxyCreator
的方法
好比一个UserServiceImp类有@service注解,而且有切点Aspectj注解加强方法,bean工厂建立userServiceImp后,代理拦截器检测到AOP相关注解,会建立动态代理对象userServiceImp$$EnhancerBySpringCGLIB并返代理对象,而不是返回userServiceImp
Spring工厂部分bean建立拦截代码逻辑
// org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.initializeBean(String, Object, RootBeanDefinition) // bean初始化 protected Object initializeBean(String beanName, Object bean, @Nullable RootBeanDefinition mbd) { invokeAwareMethods(beanName, bean); Object wrappedBean = bean; if (mbd == null || !mbd.isSynthetic()) { // 初始化以前,拦截 wrappedBean = applyBeanPostProcessorsBeforeInitialization(wrappedBean, beanName); } invokeInitMethods(beanName, wrappedBean, mbd); if (mbd == null || !mbd.isSynthetic()) { // 初始化以后拦截 wrappedBean = applyBeanPostProcessorsAfterInitialization(wrappedBean, beanName); } return wrappedBean; } @Override public Object applyBeanPostProcessorsAfterInitialization(Object existingBean, String beanName) throws BeansException { Object result = existingBean; for (BeanPostProcessor processor : getBeanPostProcessors()) { // 循环bean发布处理器调用postProcessAfterInitialization方法 Object current = processor.postProcessAfterInitialization(result, beanName); if (current == null) { return result; } result = current; } return result; }
AbstractAutoProxyCreator在此循环中被调用,好比在userServiceImp服务类上有事务注解@Transactional,通常就会被拦截生成代理对象,添加额外的处理事务的功能代码,返回加强的代理对象
默认tomcat服务器的建立就是此方法完成,此处定义特别的bean建立,通常是服务器有关或个性化对象,
//// 3.4 初始化一些与上下文有特别关系的bean对象(此处启动tomcat服务器) onRefresh(); // org.springframework.boot.web.servlet.context.ServletWebServerApplicationContext // 子类context重写 @Override protected void onRefresh() { super.onRefresh(); try { createWebServer(); //建立tomcat服务器 } catch (Throwable ex) { throw new ApplicationContextException("Unable to start web server", ex); } } private void createWebServer() { WebServer webServer = this.webServer; ServletContext servletContext = getServletContext(); if (webServer == null && servletContext == null) { ServletWebServerFactory factory = getWebServerFactory(); this.webServer = factory.getWebServer(getSelfInitializer()); // 此处建立了服务器 // org.springframework.boot.web.embedded.tomcat.TomcatServletWebServerFactory.getWebServer() } // ...... initPropertySources(); }
服务器启动后,建立spring工厂里面缓存的bean信息(没有被建立的单例)
//// 3.5 实例化全部bean工厂缓存的bean对象(剩下的). finishBeanFactoryInitialization(beanFactory); protected void finishBeanFactoryInitialization(ConfigurableListableBeanFactory beanFactory) { // ...... // Instantiate all remaining (non-lazy-init) singletons. beanFactory.preInstantiateSingletons(); } // 子类org.springframework.beans.factory.support.DefaultListableBeanFactory实现方法,完成剩下的单例bean对象建立 @Override public void preInstantiateSingletons() throws BeansException { List<String> beanNames = new ArrayList<>(this.beanDefinitionNames); for (String beanName : beanNames) { RootBeanDefinition bd = getMergedLocalBeanDefinition(beanName); if (!bd.isAbstract() && bd.isSingleton() && !bd.isLazyInit()) { getBean(beanName); //建立尚未实例化的bean对象 } } }
上下文初始化完成以后,启动tomcat服务器
finishRefresh(); // super.finishRefresh protected void finishRefresh() { // ...... 发布刷行完成事件 // Publish the final event. publishEvent(new ContextRefreshedEvent(this)); } // org.springframework.boot.web.servlet.context.ServletWebServerApplicationContext.finishRefresh() @Override protected void finishRefresh() { super.finishRefresh(); WebServer webServer = startWebServer();// 启动服务器 if (webServer != null) { publishEvent(new ServletWebServerInitializedEvent(webServer, this)); } }
发布通知监听器启动完成,监听器会根据事件类型作个性化操做
listeners.started(context); listeners.running(context); void started(ConfigurableApplicationContext context) { for (SpringApplicationRunListener listener : this.listeners) { listener.started(context); } } void running(ConfigurableApplicationContext context) { for (SpringApplicationRunListener listener : this.listeners) { listener.running(context); } } @Override public void started(ConfigurableApplicationContext context) { context.publishEvent(new ApplicationStartedEvent(this.application, this.args, context)); } @Override public void running(ConfigurableApplicationContext context) { context.publishEvent(new ApplicationReadyEvent(this.application, this.args, context)); }
不按期更新...