Spring bean加载顺序

最近在开发过程中遇到一个Spring bean加载顺序的问题,容器在启动的时候 会通过InitializingBean 的afterPropertiesSet 加载两级缓存,但是加载的过程中因为没有将load的容器对象加载到Spring 容器中导致NPE ,所以今天来巩固一下spring bean的加载顺序会受那些影响

在默认配置下,Spring的Bean加载顺序并不是随机的,但也不是严格按某种预定顺序进行的。默认情况下,Spring会按以下策略来加载和初始化Bean:

1. 加载顺序的确定性

  • 配置顺序:在Spring XML配置文件或Java配置类中定义的Bean,通常按照定义的顺序加载。
  • 注解扫描顺序:在使用组件扫描(例如@ComponentScan)时,Spring会按照类路径扫描的顺序加载Bean。但这个顺序在不同的运行环境下可能有所不同,因为类路径扫描的顺序取决于文件系统或JAR包的排列方式。

    2. 依赖关系的解析

    Spring会在加载和初始化Bean时解析Bean之间的依赖关系,确保依赖的Bean先加载和初始化。这意味着如果Bean A依赖于Bean B,那么Bean B会先于Bean A加载和初始化。

    3. Bean初始化顺序

    • @DependsOn注解:可以使用@DependsOn注解明确指定一个Bean依赖于另外一个或多个Bean。这样,被依赖的Bean会先初始化。
    • depends-on属性:在XML配置中,可以使用depends-on属性明确指定一个Bean依赖于另外一个或多个Bean。

      4. FactoryBean

      FactoryBean会优先于普通Bean进行初始化,因为它们负责创建其他Bean的实例。

      5. 生命周期回调方法

      Spring会确保Bean按照以下顺序进行生命周期回调:

      • BeanPostProcessor的postProcessBeforeInitialization方法
      • 初始化回调(例如InitializingBean的afterPropertiesSet方法或自定义init-method)
      • BeanPostProcessor的postProcessAfterInitialization方法

        示例代码和说明

        以下是一些示例代码,展示了不同情况下的Bean加载顺序:

        XML配置

        在这种情况下,beanA会先于beanB加载和初始化。

        注解配置
        @Configuration
        @ComponentScan(basePackages = "com.example")
        public class AppConfig { @Bean
            @DependsOn("beanA")
            public BeanB beanB() { return new BeanB();
            }
            @Bean
            public BeanA beanA() { return new BeanA();
            }
        }
        

        在这种配置中,beanA会先于beanB加载和初始化,因为beanB使用了@DependsOn注解。

        检查Bean加载顺序的代码示例

        为了演示Spring Bean加载的顺序,我们可以编写一个简单的Spring应用,并在Bean的构造方法中打印日志:

        @Component
        public class BeanA { public BeanA() { System.out.println("BeanA instantiated");
            }
        }
        @Component
        public class BeanB { public BeanB() { System.out.println("BeanB instantiated");
            }
        }
        @Configuration
        @ComponentScan(basePackages = "com.example")
        public class AppConfig { public static void main(String[] args) { ApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class);
                // Force initialization of beans
                context.getBean(BeanA.class);
                context.getBean(BeanB.class);
            }
        }
        

        在这个示例中,如果运行程序,控制台输出的顺序将会表明Bean的加载顺序。这种方式可以帮助我们验证在不同配置和环境下Bean的加载顺序。

        总结

        在默认配置下,Spring的Bean加载顺序主要取决于Bean的定义顺序、依赖关系和生命周期回调方法。虽然类路径扫描的顺序可能因运行环境而异,但通过使用@DependsOn注解和depends-on属性,可以显式控制Bean的加载和初始化顺序。Spring并不会随机改变Bean的加载顺序,而是遵循上述策略确保Bean在合理的顺序中加载和初始化。