springboot ApplicationContextInitializer的三种使用方法小结

目录
  • ApplicationContextInitializer的三种使用方法
    • 概述
    • 1、使用spring.factories方式
    • 2、application.properties添加配置方式
    • 3、直接通过add方法
  • ApplicationContextInitializer都干了些什么

ApplicationContextInitializer的三种使用方法

概述

ApplicationContextInitializer是在springboot启动过程(refresh方法前)调用,主要是在ApplicationContextInitializer中initialize方法中拉起了ConfigurationClassPostProcessor这个类(我在springboot启动流程中有描述),通过这个processor实现了beandefinition。

言归正传,ApplicationContextInitializer实现主要有3中方式:

1、使用spring.factories方式

首先我们自定义个类实现了ApplicationContextInitializer,然后在resource下面新建/META-INF/spring.factories文件。


public class Demo01ApplicationContextInitializer implements ApplicationContextInitializer<ConfigurableApplicationContext> {
    @Override
    public void initialize(ConfigurableApplicationContext configurableApplicationContext) {
        System.out.println("user add method ==> ApplicationContextInitializer");
    }
}

这个加载过程是在SpringApplication中的getSpringFactoriesInstances()方法中直接加载并实例后执行对应的initialize方法。代码如下:

private <T> Collection<? extends T> getSpringFactoriesInstances(Class<T> type,
			Class<?>[] parameterTypes, Object... args) {
		ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
		// Use names and ensure unique to protect against duplicates
		Set<String> names = new LinkedHashSet<String>(
				SpringFactoriesLoader.loadFactoryNames(type, classLoader));
		List<T> instances = createSpringFactoriesInstances(type, parameterTypes,
				classLoader, args, names);
		AnnotationAwareOrderComparator.sort(instances);
		return instances;
	}

2、application.properties添加配置方式

对于这种方式是通过DelegatingApplicationContextInitializer这个初始化类中的initialize方法获取到application.properties中context.initializer.classes对应的类并执行对应的initialize方法。

只需要将实现了ApplicationContextInitializer的类添加到application.properties即可。如下:

下面我们看看DelegatingApplicationContextInitializer是如何加载的。看代码:

private static final String PROPERTY_NAME = "context.initializer.classes";
private List<Class<?>> getInitializerClasses(ConfigurableEnvironment env) {
		String classNames = env.getProperty(PROPERTY_NAME);
		List<Class<?>> classes = new ArrayList<Class<?>>();
		if (StringUtils.hasLength(classNames)) {
			for (String className : StringUtils.tokenizeToStringArray(classNames, ",")) {
				classes.add(getInitializerClass(className));
			}
		}
		return classes;
	}

是从配置文件中获取到对应的初始化类信息,然后执行初始化方法。

3、直接通过add方法

这个方法就比较简单,直接在springboot启动的时候,add一个实现了ApplicationContextInitializer的类即可,代码如下:

@SpringBootApplication
public class InitializerDemoApplication {
	public static void main(String[] args) {
		//type01
		SpringApplication springApplication = new SpringApplication(InitializerDemoApplication.class);
		springApplication.addInitializers(new Demo01ApplicationContextInitializer());
		springApplication.run(args);

		//SpringApplication.run(InitializerDemoApplication.class,args);
	}
}

以上3中方法都可以实现自定义的Initializer,只不过执行的顺序有差异。这里我比较感兴趣有2个,一个通过spring.factories实现SPI模式,有兴趣的可以看下jdbc-starter等一些相关springboot starter。

第二个就是作为一个钩子去拉起来"一坨"的bean。

ApplicationContextInitializer都干了些什么

初始化方法

org.springframework.boot.context.config.DelegatingApplicationContextInitializer

 @Override
 public void initialize(ConfigurableApplicationContext context) {
  ConfigurableEnvironment environment = context.getEnvironment();
  /**
  * 初始化环境变量中的context.initializer.classes指定的类
  **/
  List<Class<?>> initializerClasses = getInitializerClasses(environment);
  if (!initializerClasses.isEmpty()) {
   applyInitializerClasses(context, initializerClasses);
  }
 }

也就是说没有定义的话,就不会初始化了。

org.springframework.boot.autoconfigure.SharedMetadataReaderFactoryContextInitializer

 @Override
 public void initialize(ConfigurableApplicationContext applicationContext) {
  /**
  * 注册一个元数据读取的工厂类
  **/
  applicationContext.addBeanFactoryPostProcessor(new CachingMetadataReaderFactoryPostProcessor());
 }

org.springframework.boot.context.ContextIdApplicationContextInitializer

 @Override
 public void initialize(ConfigurableApplicationContext applicationContext) {
  ContextId contextId = getContextId(applicationContext);
  applicationContext.setId(contextId.getId());
  /**
  * 注册一个ContextId对象
  **/
  applicationContext.getBeanFactory().registerSingleton(ContextId.class.getName(), contextId);
 }

org.springframework.boot.context.ConfigurationWarningsApplicationContextInitializer

 @Override
 public void initialize(ConfigurableApplicationContext context) {
  /**
  * 注入ComponentScan检查处理对象
  **/
  context.addBeanFactoryPostProcessor(new ConfigurationWarningsPostProcessor(getChecks()));
 }

org.springframework.boot.rsocket.context.RSocketPortInfoApplicationContextInitializer

 @Override
 public void initialize(ConfigurableApplicationContext applicationContext) {
   /**
  * 注入一个端口检查和设置的监听器,对应的事件RSocketServerInitializedEvent
  **/
  applicationContext.addApplicationListener(new Listener(applicationContext));
 }

org.springframework.boot.web.context.ServerPortInfoApplicationContextInitializer

 @Override
 public void initialize(ConfigurableApplicationContext applicationContext) {
  /**
   注册了一个监听org.springframework.boot.web.context.WebServerInitializedEvent事件的监听器,用于设置端口信息
  **/
  applicationContext.addApplicationListener(this);
 }

org.springframework.boot.autoconfigure.logging.ConditionEvaluationReportLoggingListener

 @Override
 public void initialize(ConfigurableApplicationContext applicationContext) {
  this.applicationContext = applicationContext;
  applicationContext.addApplicationListener(new ConditionEvaluationReportListener());
  if (applicationContext instanceof GenericApplicationContext) {
   // Get the report early in case the context fails to load
   // 注册一个监听ApplicationEvent事件的监听器用于打印自动配置后的日志信息
   this.report = ConditionEvaluationReport.get(this.applicationContext.getBeanFactory());
  }
 }

所有的这些初始化类都没偶进行启动服务的实质性操作,都是通过注册对象,埋点,后面invokeBeanFactoryPostProcessors才真正调用初始化方法,而且在项目启动之前

以上为个人经验,希望能给大家一个参考,也希望大家多多支持我们。

(0)

相关推荐

  • SpringBoot上下文初始器加载过程详解

    利用 Spring 工厂加载机制,实例化 ApplicationContextInitializer 实现类,并排序对象集合. 关键方法 private <T> Collection<T> getSpringFactoriesInstances(Class<T> type, Class<?>[] parameterTypes, Object... args) { ClassLoader classLoader = Thread.currentThread()

  • Springboot启动扩展点超详细教程小结

    1.背景 Spring的核心思想就是容器,当容器refresh的时候,外部看上去风平浪静,其实内部则是一片惊涛骇浪,汪洋一片.Springboot更是封装了Spring,遵循约定大于配置,加上自动装配的机制.很多时候我们只要引用了一个依赖,几乎是零配置就能完成一个功能的装配. 我非常喜欢这种自动装配的机制,所以在自己开发中间件和公共依赖工具的时候也会用到这个特性.让使用者以最小的代价接入.想要把自动装配玩的转,就必须要了解spring对于bean的构造生命周期以及各个扩展接口.当然了解了bean

  • SpringBoot整个启动过程的分析

    前言 前一篇分析了SpringBoot如何启动以及内置web容器,这篇我们一起看一下SpringBoot的整个启动过程,废话不多说,正文开始. 正文 一.SpringBoot的启动类是**application,以注解@SpringBootApplication注明. @SpringBootApplication public class CmsApplication { public static void main(String[] args) { SpringApplication.run

  • springboot ApplicationContextInitializer的三种使用方法小结

    目录 ApplicationContextInitializer的三种使用方法 概述 1.使用spring.factories方式 2.application.properties添加配置方式 3.直接通过add方法 ApplicationContextInitializer都干了些什么 ApplicationContextInitializer的三种使用方法 概述 ApplicationContextInitializer是在springboot启动过程(refresh方法前)调用,主要是在A

  • SQL中的三种去重方法小结

    目录 distinct group by row_number 在使用SQL提数的时候,常会遇到表内有重复值的时候,比如我们想得到 uv (独立访客),就需要做去重. 在 MySQL 中通常是使用 distinct 或 group by子句,但在支持窗口函数的 sql(如Hive SQL.Oracle等等) 中还可以使用 row_number 窗口函数进行去重. 举个栗子,现有这样一张表 task: task_id order_id start_time 1 123 2020-01-05 1 2

  • MyBatis-Plus非表字段的三种处理方法小结

    目录 MyBatis-Plus非表字段的处理方式 1.使用关键字 2.关键字static 3.使用@TableField注解 mybatisplus 非数据库字段设置 使用transient修饰 MyBatis-Plus非表字段的处理方式 1.使用关键字 transient修饰字段,不让这个字段参与序列化 //备注 private transient String remark; 2.关键字static mybatis不会给static构建setter和getter,需要自己构建setter和g

  • springboot集成本地缓存Caffeine的三种使用方式(小结)

    目录 第一种方式(只使用Caffeine) 第二种方式(使用Caffeine和spring cache) 第三种方式(使用Caffeine和spring cache) 第一种方式(只使用Caffeine) gradle添加依赖 dependencies { implementation 'org.springframework.boot:spring-boot-starter-jdbc' implementation 'org.springframework.boot:spring-boot-s

  • C#多态的三种实现方式(小结)

    C#实现多态主要有3种方法,虚方法,抽象类,接口 1 虚方法 在父类的方法前面加关键字virtual, 子类重写该方法时在方法名前面加上override关键字,例如下面的Person类的SayHello方法 class Person { public Person(string name) { this.Name = name; } string _name; public string Name { get => _name; set => _name = value; } //父类方法加v

  • Python中浅拷贝的四种实现方法小结

    目录 方式一:使用切片 [:] 方式二:使用工厂函数 方式三:使用数据类型自带的 copy 方法 方式四:使用 copy 模块的 copy 方法 方式一:使用切片 [:] 列表 # 浅拷贝 [:] old_list = [1, 2, [3, 4]] new_list = old_list[:] old_list.append(5) old_list[2][0] += 97 print("Old list:", old_list, "old list id:", id

  • Python time三种时间转换小结

    目录 复习回顾 1. time模块源代码描述 2. time模块核心对象 3. 时间戳与结构化时间转换 4. 字符串时间与结构化时间转换 总结 复习回顾 我们已经对Python内置模块-time中知道时间格式目前有三种. 时间戳 结构化时间 字符串时间 本期,我们将继续深入对time模块中所涉及的三种时间进行学习,Let's go~ 1. time模块源代码描述 我们通过import time导入time后,可以通过长按CTRL键后查看到time.pyi文件查看time.pyi文件,我们会得知如

  • C++中strlen函数的三种实现方法

    目录 一.strlen函数是什么 二.strlen的三种实现方法 1.第一种方法(直接) 2.第二种方法(递归) 3.第三种方法(指针-指针) 四.小结 一.strlen函数是什么 我们经常用到strlen这个函数求字符串长度,但是它是怎么实现的呢?接下来让给我用三种方法带你们看看它是如何实现? 首先我们先来了解一下strlen这个函数,strlen 是求字符串长度的函数,它的返回值是size_t,就是unsigned int.字符串以'\0'作为结束标志,strlen函数返回的值就是在字符串中

  • JS面向对象(3)之Object类,静态属性,闭包,私有属性, call和apply的使用,继承的三种实现方法

    1.Object类 在JS中,Object是所有类的基类,使用Object类来创建自定义对象时,可以无需定义构造函数(constructor,prototype,hasOwnProperty(property)) var per = new Object(); per.name = 'zhangsan'; per.age = ; alert(per.name + per.age); 我们想在程序中得到一个对象变量,只要能存储大量数据即可,这个时候,我们可以考虑使用Object类.Object类避

  • C语言中函数指针的三种使用方法总结

     C语言中函数指针的三种使用方法总结 在这里分享一下自己的心得,希望和大家一起分享技术,如果有什么不足,还请大家指正.写出这篇目的,就是希望大家一起成长,我也相信技术之间没有高低,只有互补,只有分享,才能使彼此更加成长. 定义方式:int (*p)(int x, int y); 实现代码: #include <stdio.h> int sum(int x, int y){ return x + y; } int reduce(int x, int y){ return x - y; } int

随机推荐