springboot中bean的加载顺序

springboot中bean的加载顺序

一、为什么要控制

当你在项目启动时需要提前做一个业务的初始化工作时,或者你正在开发某个中间件需要完成自动装配时。你会声明自己的Configuration类,但是可能你面对的是好几个有互相依赖的Bean。如果不加以控制,这时候可能会报找不到依赖的错误,这个时候需要通过一些手段来控制springboot中的bean加载顺序。

二、怎么控制

@DependsOn

@DependsOn注解可以用来控制bean的创建顺序,该注解用于声明当前bean依赖于另外一个bean。所依赖的bean会被容器确保在当前bean实例化之前被实例化。与@Component或@Bean配合使用
demo

@Slf4j
@Configuration
@ConfigurationProperties(prefix = "dict")
public class SpringConfig {
    @Component(value = "EventSource")
    public class EventSource {
        public EventSource(){
            System.out.println("事件源创建");
        }
    }
    /**
     * 监听类
     */
    @Component
    @DependsOn(value = {"EventSource"})
    public class EventTListener {

        public EventTListener(){
            System.out.println("监听器创建");
        }
    }
}

在这里插入图片描述

参数注入

package com.sinosoft.springbootplus.test.config;

import lombok.extern.slf4j.Slf4j;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.DependsOn;
import org.springframework.stereotype.Component;

/**
 * @author lsh
 * @date 2022/2/25
 */
@Slf4j
@Configuration
@ConfigurationProperties(prefix = "dict")
public class SpringConfig {
    @Component
    public class Event{
        public Event(){
            System.out.println("事件事件");
        }
    }
    
    @Component
    public class EventSource{
        public EventSource(Event e){
            System.out.println("事件源创建");
        }
    }


    @Component
    public class EventTListener {

        public EventTListener(){
            System.out.println("监听器创建");
        }
    }
}

在这里插入图片描述

利用bean的生命周期中的扩展点

@AutoConfigureOrder

@AutoConfigureOrder只能改变外部依赖的@Configuration的顺序。
这是不对的用法

@Slf4j
@Configuration
@ConfigurationProperties(prefix = "dict")
public class SpringConfig {
    @Component
    @AutoConfigureOrder(1)
    public class Event{
        public Event(){
            System.out.println("事件事件");
        }
    }

    @Component
    @AutoConfigureOrder(2)
    public class EventSource{
        public EventSource(Event e){
            System.out.println("事件源创建");
        }
    }


    @Component
    @AutoConfigureOrder(3)
    public class EventTListener {

        public EventTListener(){
            System.out.println("监听器创建");
        }
    }
}

在这里插入图片描述
以上内容发现,在config里配置是不起作用的。
这是正确的用法

  1. 创建两个配置类
@Slf4j
@Configuration
@AutoConfigureOrder(1)
public class SpringConfig {
    @Component
    public class Event{
        public Event(){
            System.out.println("首先在SpringConfig");
        }
    }

}


@Slf4j
@Configuration
@AutoConfigureOrder(2)
public class NewConfig {
    @Component
    public class Event{
        public Event(){
            System.out.println("然后在NewConfig");
        }
    }
}
  1. 测试
    在这里插入图片描述
    发现结果是不正确的,注解还是没有生效。
  2. 当前工程里增加配置 META-INF/spring.factories,内容为项目中的配置类
org.springframework.boot.autoconfigure.EnableAutoConfiguration=com.sinosoft.springbootplus.common.config.NewConfig,com.sinosoft.springbootplus.common.config.SpringConfig

  1. 测试结果如图(正确)
    在这里插入图片描述

三、遇到的问题:

1、需要根据配置决定生成哪个实现类。

在这里插入图片描述
当在配置文件中配置的dict.cacheType的值是local时,初始化LocalISysDictRepository交给spring容器管理;当项目依赖了redis并且配置文件中配置的dict.cacheType的值是redis时,初始化RedisISysDictRepository交给spring容器管理。
2、但是我又在这两个实现类上加了@Repository注解,也要交给Spring管理,这个时候项目启动就报错了。(通俗的来说一个类只能一次交给Spring管理)
在这里插入图片描述

  • 1
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值