深入理解Spring bean加载顺序
作者:jeff-y
最近在开发过程中遇到一个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配置
<bean id="beanA" class="com.example.BeanA"/> <bean id="beanB" class="com.example.BeanB" depends-on="beanA"/>
在这种情况下,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在合理的顺序中加载和初始化。
到此这篇关于深入理解Spring bean加载顺序的文章就介绍到这了,更多相关Spring bean加载顺序内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!