使用Spring @DependsOn控制bean加載順序


使用Spring @DependsOn控制bean加載順序

spring容器載入bean順序是不確定的,spring框架沒有約定特定順序邏輯規范。但spring保證如果A依賴B(如beanA中有@Autowired B的變量),那么B將先於A被加載。但如果beanA不直接依賴B,我們如何讓B仍先加載呢?

控制bean初始化順序

可能有些場景中,bean A 間接依賴 bean B。如Bean B應該需要更新一些全局緩存,可能通過單例模式實現且沒有在spring容器注冊,bean A需要使用該緩存;因此,如果bean B沒有准備好,bean A無法訪問。

另一個場景中,bean A是事件發布者(或JMS發布者),bean B (或一些) 負責監聽這些事件,典型的如觀察者模式。我們不想B 錯過任何事件,那么B需要首先被初始化。

簡言之,有很多場景需要bean B應該被先於bean A被初始化,從而避免各種負面影響。我們可以在bean A上使用@DependsOn注解,告訴容器bean B應該先被初始化。下面通過示例來說明。

示例說明

示例通過事件機制說明,發布者和監聽者,然后通過spring配置運行。為了方便說明,示例進行了簡化。

EventManager.java
事件管理類,維護監聽器列表,通過單例方法獲取事件管理器,可以增加監聽器或發布事件。

import java.util.ArrayList;
import java.util.List;
import java.util.function.Consumer;

public class EventManager {
    private final List<Consumer<String>> listeners = new ArrayList<>();

    private EventManager() {
    }

    private static class SingletonHolder {
        private static final EventManager INSTANCE = new EventManager();
    }

    public static EventManager getInstance() {
        return SingletonHolder.INSTANCE;
    }

    public void publish(final String message) {
        listeners.forEach(l -> l.accept(message));
    }

    public void addListener(Consumer<String> eventConsumer) {
        listeners.add(eventConsumer);
    }
}

EventPublisherBean.java
事件發布類,通過EventManager類發布事件。

import com.logicbig.example.EventManager;

public class EventPublisherBean {

    public void initialize() {
        System.out.println("EventPublisherBean initializing");
        EventManager.getInstance().publish("event published from EventPublisherBean");
    }
}

EventListenerBean.java
事件監聽者,可以增加監聽器。

import com.logicbig.example.EventManager;

public class EventListenerBean {

    private void initialize() {
        EventManager.getInstance().
                addListener(s ->
                        System.out.println("event received in EventListenerBean : " + s));
    }
}

AppConfig.java
配置運行類。

@Configuration
@ComponentScan("com.logicbig.example")
public class AppConfig {

    @Bean(initMethod = "initialize")
    @DependsOn("eventListener")
    public EventPublisherBean eventPublisherBean () {
        return new EventPublisherBean();
    }

    @Bean(name = "eventListener", initMethod = "initialize")
    // @Lazy
    public EventListenerBean eventListenerBean () {
        return new EventListenerBean();
    }

    public static void main (String... strings) {
        new AnnotationConfigApplicationContext(AppConfig.class);
    }
}

運行AppConfig的main方法,輸出結果為:

EventListenerBean initializing
EventPublisherBean initializing
event received in EventListenerBean : event published from EventPublisherBean

總結

如果我們注釋掉@DependsOn("eventListener"),我們可能不確定獲得相同結果。嘗試多次運行main方法,偶爾我們將看到EventListenerBean 沒有收到事件。為什么是偶爾呢?因為容器啟動過程中,spring按任意順序加載bean。

那么當不使用@DependsOn可以讓其100%確定嗎?可以使用@Lazy注解放在eventListenerBean ()上。因為EventListenerBean在啟動階段不加載,當其他bean需要其時才加載。這次我們僅EventListenerBean被初始化

EventPublisherBean initializing
現在從新增加@DependsOn,也不刪除@Lazy注解,輸出結果和第一次一致,雖然我們使用了@Lazy注解,eventListenerBean在啟動時仍然被加載,因為@DependsOn表明需要EventListenerBean。

 

@DependsOn的默認值是數組,那么我們可以寫成這樣
@DependsOn(value= {"b","c","d"})
public class A {
    
}

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM