启动入口

@SpringBootApplication
public class DemoApplication {
    public static void main(String[] args) {
        // step.1
        SpringApplication.run(DemoApplication.class, args);
    }
}

实例化 SpringApplication 对象

// step.2
public static ConfigurableApplicationContext run(Class<?> primarySource, String... args) {
    return run(new Class[]{primarySource}, args);
}

// step.3 
// 实例化SpringApplication对象,在构造方法内加载一些资源
public static ConfigurableApplicationContext run(Class<?>[] primarySources, String[] args) {
    return new SpringApplication(primarySources).run(args);
}

// step.4
public SpringApplication(Class<?>... primarySources) {
    this(null, primarySources);
}

加载运行类信息 & 确定应用类型 & 加载初始化器 & 加载监听器 & 设置运行主类

// step.5
@SuppressWarnings({ "unchecked", "rawtypes" })
public SpringApplication(ResourceLoader resourceLoader, Class<?>... primarySources) {
    this.resourceLoader = resourceLoader;
    Assert.notNull(primarySources, "PrimarySources must not be null");
    // 加载运行类信息,此处的primarySources实际上是启动传入的class
    this.primarySources = new LinkedHashSet<>(Arrays.asList(primarySources));
    // 确定应用类型,默认servlet
    this.webApplicationType = WebApplicationType.deduceFromClasspath();
    this.bootstrapRegistryInitializers = new ArrayList<>(getSpringFactoriesInstances(BootstrapRegistryInitializer.class));
    // 加载初始化器
    setInitializers((Collection) getSpringFactoriesInstances(ApplicationContextInitializer.class));
    // 加载监听器
    setListeners((Collection)getSpringFactoriesInstances(ApplicationListener.class));
    // 设置运行主类
    this.mainApplicationClass = deduceMainApplicationClass();
}

从 getSpringFactoriesInstances() 方法进入可以发现,系统使用 SpringFactoriesLoader 来加载一些预设的配置

  • SpringFactoriesLoader 正是 SpringBoot 简化配置的方式

FACTORIES_RESOURCE_LOCATION 由此静态常量定义可知,SpringFactoriesLoader 会将此目录下定义的一些配置加载进来,且支持扩展,开发者根据引入规则按需配置即可,无需再像以前配置spring时写xml文件

public final class SpringFactoriesLoader {

/**
 * The location to look for factories.
 * <p>Can be present in multiple JAR files.
 */
public static final String FACTORIES_RESOURCE_LOCATION = "META-INF/spring.factories";


private static final Log logger = LogFactory.getLog(SpringFactoriesLoader.class);

static final Map<ClassLoader, Map<String, List<String>>> cache = new ConcurrentReferenceHashMap<>();


private SpringFactoriesLoader() {
}

如果希望自定义初始化器,则需要实现 ApplicationContextInitializer 接口
加载希望自定义监听器,则需要实现 ApplicationListener 接口
以上两者都需要在resources目录下添加 META-INF/spring.factories 配置文件,将自定义的初始化器注册进去
image.png

执行启动方法 run()

// step.6
/**
 * Run the Spring application, creating and refreshing a new
 * {@link ApplicationContext}.
 * @param args the application arguments (usually passed from a Java main method)
 * @return a running {@link ApplicationContext}
 */
public ConfigurableApplicationContext run(String... args) {
    // 记录启动开始时间
    long startTime = System.nanoTime();
    // 创建引导上下文
    DefaultBootstrapContext bootstrapContext = createBootstrapContext();
    // 声明应用上下文
    ConfigurableApplicationContext context = null;
    // 设置 无显示器依然启动
    configureHeadlessProperty();
    // 创建所有 Spring 运行监听器并发布应用启动事件 并启用监听器
    SpringApplicationRunListeners listeners = getRunListeners(args);
    listeners.starting(bootstrapContext, this.mainApplicationClass);
    try {
        // 封装命令行参数
        ApplicationArguments applicationArguments = new DefaultApplicationArguments(args);
        // 环境准备 加载当前指定环境的运行参数 例如application-dev.yml 或 .properties
        ConfigurableEnvironment environment = prepareEnvironment(listeners, bootstrapContext, applicationArguments);
        // 设置需要忽略的bean
        configureIgnoreBeanInfo(environment);
        // 打印banner 可通过 SpringApplication.setBannerMode(Banner.Mode.OFF)
        Banner printedBanner = printBanner(environment);
        // 创建应用上下文 即spring容器
        context = createApplicationContext();
        context.setApplicationStartup(this.applicationStartup);
        // 设置bean工厂;初始化.factories中配置;各监听器处理上下文准备事件;设置bean定义加载器;设置资源加载器;设置beanName生成器等
        prepareContext(bootstrapContext, context, environment, listeners, applicationArguments, printedBanner);
        // 设置容器状态;给bean工厂设置;添加后置处理器;国际化;派发早期事件;创建web容器;将上述流程中被加载的bean实例化等
        refreshContext(context);
        // 刷新上下文后置处理
        afterRefresh(context, applicationArguments);
        // 记录启动完成时长
        Duration timeTakenToStartup = Duration.ofNanos(System.nanoTime() - startTime);
        if (this.logStartupInfo) {
            new StartupInfoLogger(this.mainApplicationClass).logStarted(getApplicationLog(), timeTakenToStartup);
        }
        // 发布上下文准备就绪事件
        listeners.started(context, timeTakenToStartup);
        // 执行自定义的run方法
        callRunners(context, applicationArguments);
    }
    catch (Throwable ex) {
        handleRunFailure(context, ex, listeners);
        throw new IllegalStateException(ex);
    }
    try {
        Duration timeTakenToReady = Duration.ofNanos(System.nanoTime() - startTime);
        listeners.ready(context, timeTakenToReady);
    }
    catch (Throwable ex) {
        handleRunFailure(context, ex, null);
        throw new IllegalStateException(ex);
    }
    return context;
}
  • 创建引导上下文
    可以看到在默认情况下并没有bootstrapRegistryInitializers,个人理解为此处支持了开发者自定义某些在spring容器创建前需要执行的初始化器
    image.png
  • 设置环境变量
    可以看到此处加载了一些自定义的环境配置文件
    image.png
  • 创建Spring容器
    image.png
  • 运行时监听器
    image.png

扩展功能 callRunners(context, applicationArguments)
可以在启动完成后执行自定义的内容;有2种实现方式

  • 实现 ApplicationRunner 接口
  • 实现 CommandLineRunner 接口

参考资料:
https://blog.csdn.net/weixin_...
https://blog.csdn.net/w200921...


老污的猫
30 声望5 粉丝