中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

SpringBoot 初始化之7招式

發布時間:2020-06-11 15:39:58 來源:網絡 閱讀:4706 作者:美碼師 欄目:編程語言

SpringBoot 初始化之7招式

背景

在日常開發時,我們常常需要 在SpringBoot 應用啟動時執行某一段邏輯,如下面的場景:

  • 獲取一些當前環境的配置或變量
  • 向數據庫寫入一些初始數據
  • 連接某些第三方系統,確認對方可以工作..

在實現這些功能時,我們可能會遇到一些"坑"。 為了利用SpringBoot框架的便利性,我們不得不將整個應用的執行控制權交給容器,于是造成了大家對于細節是一無所知的。
那么在實現初始化邏輯代碼時就需要小心了,比如,我們并不能簡單的將初始化邏輯在Bean類的構造方法中實現,類似下面的代碼:

@Component
public class InvalidInitExampleBean {

    @Autowired
    private Environment env;

    public InvalidInitExampleBean() {
        env.getActiveProfiles();
    }
}

這里,我們在InvalidInitExampleBean的構造方法中試圖訪問一個自動注入的env字段,當真正執行時,你一定會得到一個空指針異常(NullPointerException)。
原因在于,當構造方法被調用時,Spring上下文中的Environment這個Bean很可能還沒有被實例化,同時也仍未注入到當前對象,所以并不能這樣進行調用。

下面,我們來看看在SpringBoot中實現"安全初始化"的一些方法:

1、 @PostConstruct 注解

@PostConstruct 注解其實是來自于 javax的擴展包中(大多數人的印象中是來自于Spring框架),它的作用在于聲明一個Bean對象初始化完成后執行的方法
來看看它的原始定義:

The PostConstruct annotation is used on a method that needs to be executed after dependency injection is done to perform any initialization.

也就是說,該方法會在所有依賴字段注入后才執行,當然這一動作也是由Spring框架執行的。

下面的代碼演示了使用@PostConstruct的例子:

@Component
public class PostConstructExampleBean {

    private static final Logger LOG 
      = Logger.getLogger(PostConstructExampleBean.class);

    @Autowired
    private Environment environment;

    @PostConstruct
    public void init() {
        //environment 已經注入
        LOG.info(Arrays.asList(environment.getDefaultProfiles()));
    }
}

2、 InitializingBean 接口

InitializingBean 是由Spring框架提供的接口,其與@PostConstruct注解的工作原理非常類似。
如果不使用注解的話,你需要讓Bean實例繼承 InitializingBean接口,并實現afterPropertiesSet()這個方法。

下面的代碼,展示了這種用法:

@Component
public class InitializingBeanExampleBean implements InitializingBean {

    private static final Logger LOG 
      = Logger.getLogger(InitializingBeanExampleBean.class);

    @Autowired
    private Environment environment;

    @Override
    public void afterPropertiesSet() throws Exception {
        //environment 已經注入
        LOG.info(Arrays.asList(environment.getDefaultProfiles()));
    }
}

3、 @Bean initMethod方法

我們在聲明一個Bean的時候,可以同時指定一個initMethod屬性,該屬性會指向Bean的一個方法,表示在初始化后執行。

如下所示:

@Bean(initMethod="init")
public InitMethodExampleBean exBean() {
    return new InitMethodExampleBean();
}

然后,這里將initMethod指向init方法,相應的我們也需要在Bean中實現這個方法:

public class InitMethodExampleBean {

    private static final Logger LOG = Logger.getLogger(InitMethodExampleBean.class);

    @Autowired
    private Environment environment;

    public void init() {
        LOG.info(Arrays.asList(environment.getDefaultProfiles()));
    }
}

上面的代碼是基于Java注解的方式,使用Xml配置也可以達到同樣的效果:

<bean id="initMethodExampleBean"
  class="org.baeldung.startup.InitMethodExampleBean"
  init-method="init">
</bean>

該方式在早期的 Spring版本中大量被使用

4、 構造器注入

如果依賴的字段在Bean的構造方法中聲明,那么Spring框架會先實例這些字段對應的Bean,再調用當前的構造方法。
此時,構造方法中的一些操作也是安全的,如下:

@Component
public class LogicInConstructorExampleBean {

    private static final Logger LOG 
      = Logger.getLogger(LogicInConstructorExampleBean.class);

    private final Environment environment;

    @Autowired
    public LogicInConstructorExampleBean(Environment environment) {
        //environment實例已初始化
        this.environment = environment;
        LOG.info(Arrays.asList(environment.getDefaultProfiles()));
    }
}

5、 ApplicationListener

ApplicationListener 是由 spring-context組件提供的一個接口,主要是用來監聽 "容器上下文的生命周期事件"。
它的定義如下:

public interface ApplicationListener<E extends ApplicationEvent> extends EventListener {
    /**
     * Handle an application event.
     * @param event the event to respond to
     */
    void onApplicationEvent(E event);
}

這里的event可以是任何一個繼承于ApplicationEvent的事件對象。 對于初始化工作來說,我們可以通過監聽ContextRefreshedEvent這個事件來捕捉上下文初始化的時機。
如下面的代碼:

@Component
public class StartupApplicationListenerExample implements
  ApplicationListener<ContextRefreshedEvent> {

    private static final Logger LOG 
      = Logger.getLogger(StartupApplicationListenerExample.class);

    public static int counter;

    @Override public void onApplicationEvent(ContextRefreshedEvent event) {
        LOG.info("Increment counter");
        counter++;
    }
}

在Spring上下文初始化完成后,這里定義的方法將會被執行。
與前面的InitializingBean不同的是,通過ApplicationListener監聽的方式是全局性的,也就是當所有的Bean都初始化完成后才會執行方法。
Spring 4.2 之后引入了新的 @EventListener注解,可以實現同樣的效果:

@Component
public class EventListenerExampleBean {

    private static final Logger LOG 
      = Logger.getLogger(EventListenerExampleBean.class);

    public static int counter;

    @EventListener
    public void onApplicationEvent(ContextRefreshedEvent event) {
        LOG.info("Increment counter");
        counter++;
    }
}

6、 CommandLineRunner

SpringBoot 提供了一個CommanLineRunner接口,用來實現在應用啟動后的邏輯控制,其定義如下:

public interface CommandLineRunner {

    /**
     * Callback used to run the bean.
     * @param args incoming main method arguments
     * @throws Exception on error
     */
    void run(String... args) throws Exception;

}

這里的run方法會在Spring 上下文初始化完成后執行,同時會傳入應用的啟動參數。
如下面的代碼:

@Component
public class CommandLineAppStartupRunner implements CommandLineRunner {
    private static final Logger LOG =
      LoggerFactory.getLogger(CommandLineAppStartupRunner.class);

    public static int counter;

    @Override
    public void run(String...args) throws Exception {
        //上下文已初始化完成
        LOG.info("Increment counter");
        counter++;
    }
}

此外,對于多個CommandLineRunner的情況下可以使用@Order注解來控制它們的順序。

7、 ApplicationRunner

與 CommandLineRunner接口類似, Spring boot 還提供另一個ApplicationRunner 接口來實現初始化邏輯。
不同的地方在于 ApplicationRunner.run()方法接受的是封裝好的ApplicationArguments參數對象,而不是簡單的字符串參數。

@Component
public class AppStartupRunner implements ApplicationRunner {
    private static final Logger LOG =
      LoggerFactory.getLogger(AppStartupRunner.class);

    public static int counter;

    @Override
    public void run(ApplicationArguments args) throws Exception {
        LOG.info("Application started with option names : {}", 
          args.getOptionNames());
        LOG.info("Increment counter");
        counter++;
    }
}

ApplicationArguments對象提供了一些非常方便的方法,可以用來直接獲取解析后的參數,比如:

java -jar application.jar --debug --ip=xxxx

此時通過 ApplicationArguments的getOptionNames就會得到["debug","ip"]這樣的值

測試代碼

下面,通過一個小測試來演示幾種初始化方法的執行次序,按如下代碼實現一個復合式的Bean:

@Component
@Scope(value = "prototype")
public class AllStrategiesExampleBean implements InitializingBean {

    private static final Logger LOG 
      = Logger.getLogger(AllStrategiesExampleBean.class);

    public AllStrategiesExampleBean() {
        LOG.info("Constructor");
    }

    @Override
    public void afterPropertiesSet() throws Exception {
        LOG.info("InitializingBean");
    }

    @PostConstruct
    public void postConstruct() {
        LOG.info("PostConstruct");
    }

    //在XML中定義為initMethod
    public void init() {
        LOG.info("init-method");
    }
}

執行這個Bean的初始化,會發現日志輸出如下:

[main] INFO o.b.startup.AllStrategiesExampleBean - Constructor
[main] INFO o.b.startup.AllStrategiesExampleBean - PostConstruct
[main] INFO o.b.startup.AllStrategiesExampleBean - InitializingBean
[main] INFO o.b.startup.AllStrategiesExampleBean - init-method

所以,這幾種初始化的順序為:

  1. 構造器方法
  2. @PostConstruct 注解方法
  3. InitializingBean的afterPropertiesSet()
  4. Bean定義的initMethod屬性方法

參考文檔

https://www.baeldung.com/running-setup-logic-on-startup-in-spring

美碼師的 SpringBoot 補習系列

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

调兵山市| 佛冈县| 连山| 新巴尔虎左旗| 广饶县| 灵川县| 喀喇| 丰城市| 梅州市| 古交市| 海门市| 阿拉善右旗| 响水县| 巴里| 平罗县| 宾川县| 遂平县| 澄迈县| 渝北区| 木兰县| 德昌县| 南召县| 仁布县| 德化县| 攀枝花市| 都安| 临西县| 达拉特旗| 莱州市| 什邡市| 唐山市| 海兴县| 多伦县| 牡丹江市| 自治县| 华坪县| 石阡县| 秦安县| 瓮安县| 富顺县| 兴国县|