您好,登錄后才能下訂單哦!
springboot ApplicationContextInitializer的三種使用方法是怎樣的,相信很多沒有經驗的人對此束手無策,為此本文總結了問題出現的原因和解決方法,通過這篇文章希望你能解決這個問題。
ApplicationContextInitializer是在springboot啟動過程(refresh方法前)調用,主要是在ApplicationContextInitializer中initialize方法中拉起了ConfigurationClassPostProcessor這個類(我在springboot啟動流程中有描述),通過這個processor實現了beandefinition。
言歸正傳,ApplicationContextInitializer實現主要有3中方式:
首先我們自定義個類實現了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; }
對于這種方式是通過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; }
是從配置文件中獲取到對應的初始化類信息,然后執行初始化方法。
這個方法就比較簡單,直接在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。
初始化方法
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才真正調用初始化方法,而且在項目啟動之前
看完上述內容,你們掌握springboot ApplicationContextInitializer的三種使用方法是怎樣的的方法了嗎?如果還想學到更多技能或想了解更多相關內容,歡迎關注億速云行業資訊頻道,感謝各位的閱讀!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。