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是Spring框架提供的一个接口,它允许我们在Spring容器初始化之前对其进行一些自定义的配置。在应用程序启动时,Spring容器会先执行ApplicationContextInitializer中的initialize方法,然后再执行其他的配置和初始化操作。 使用ApplicationContextInitializer可以实现很多自定义的功能,比如动态读取配置文件、动态注入Bean等。下面是一个简单的代码实例: ```java public class MyApplicationContextInitializer implements ApplicationContextInitializer<ConfigurableApplicationContext> { @Override public void initialize(ConfigurableApplicationContext applicationContext) { // 在这里进行一些自定义的配置 // 比如动态读取配置文件,然后将配置信息注入到Spring容器中 // 或者动态注入Bean等 } } ``` 在上面的代码中,我们实现了ApplicationContextInitializer接口,并重写了它的initialize方法。在这个方法中,我们可以进行一些自定义的配置操作。比如动态读取配置文件,然后将配置信息注入到Spring容器中,或者动态注入Bean等。 要使用ApplicationContextInitializer,只需要在Spring配置文件中加入以下配置即可: ```xml <context-param> <param-name>contextInitializerClasses</param-name> <param-value>com.example.MyApplicationContextInitializer</param-value> </context-param> ``` 在上面的配置中,我们指定了MyApplicationContextInitializer这个类作为ApplicationContextInitializer。这样,在应用程序启动时,Spring容器就会先执行MyApplicationContextInitializer中的initialize方法,然后再执行其他的配置和初始化操作。 需要注意的是,如果有多个ApplicationContextInitializer,它们会按照配置的顺序依次执行。因此,如果有多个ApplicationContextInitializer,它们的执行顺序可能会影响应用程序的行为。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值