问题:实现了ApplicationContextAware接口后在普通类中还是获取不到bean
首先来看一下对应的例子:
TestApplicationContextAware.java
@Component
public class TestApplicationContextAware implements ApplicationContextAware {
private static ApplicationContext context;
@Override
public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
context=applicationContext;
}
public static ApplicationContext getApplicationContext(){
return context;
}
public static Object getBean(String name){
//return context.getBean(name); //为什么这样写会报空指针错误
return getApplicationContext().getBean(name);
}
public static <T> T getBean(Class<T> requiredType) throws BeansException{
return getApplicationContext().getBean(requiredType);
}
public void show() {
System.out.println("Demo3: " + context);
}
}
WebConfig.java
@Configuration
public class WebConfig {
@Bean
public User getUser(){
return new User(5,"48");
}
}
User.java
public class User {
private Integer age;
private String name;
User(Integer _age,String _name){
this.age=_age;
this.name=_name;
}
}
Client.java
public class Client {
public static void main(String[] args) {
//先需要将类纳入IOC容器管理,下面就是创建IOC容器.applicationContext 是一个应用上下文
AnnotationConfigApplicationContext applicationContext=new AnnotationConfigApplicationContext(TestApplicationContextAware.class);
//ConfigurableApplicationContext 是另外一个应用上下文
ConfigurableApplicationContext a=new AnnotationConfigApplicationContext(WebConfig.class);
System.out.println(applicationContext.getBean(TestApplicationContextAware.class));
TestApplicationContextAware testApplicationContextAware= (TestApplicationContextAware) TestApplicationContextAware.getBean("testApplicationContextAware");
//2. User user = (User) TestApplicationContextAware.getBean("getUser");
System.out.println(testApplicationContextAware.hashCode());
}
}
通过TestApplicationContextAware获取getUser这个bean时会提示: No bean named ‘getUser’ available。
原因分析
实现了ApplicationContextAware接口的类只能获得当前上下文中的bean,也就是applicationContext这个上下文。
String[] beans = applicationContext.getBeanDefinitionNames();
Arrays.sort(beans);
for (String bean : beans) {
System.out.println("applicationContext上下文管理的bean: "+bean);
}
输出:
通过上面的输出可以看到applicationContext这个上下文并没有管理getUser这个bean,所以获取不到对应的bean,只能获取到testApplicationContextAware这个bean。
记录一下自己踩坑的过程
第一次自己直接通过TestApplicationContextAware.getBean(“getUser”);去获取bean,TestApplicationContextAware.java类报错context为null,当时觉得自己已经使用注解了,类肯定也被Spring管理了。找了很久的错误,突然发现自己都没有创建IOC容器去管理bean,仅仅通过注解怎么就能让bean被Spring管理了呢。
所以后面先创建了Spring上下文管理Bean,但是通过TestApplicationContextAware.getBean(“getUser”)还是获取不到实例,经过思考要想通过IOC容器获得bean的前提该bean被IOC容器管理,但是getUser这个bean并不在applicationContext这个上下文的管控中,当然是获取不到该实例的。
学习进阶:
setApplicationContext是Spring框架预留的一个关键的钩子方法,spring详细加载全过程如下:
调用 BeanNameAware 的 setBeanName 方法
调用 BeanFactoryAware 的 setBeanFactory 方法
调用 ApplicationContextAware 的 setApplicationContext
调用 InitializingBean 的 afterPropertiesSet 或者没有实现这个接口,但指定了@Bean(initMethod=“不加括号的方法名”),会执行这个方法
调用 BeanPostProcessor 的 postProcessBeforeInitialization 方法
调用 BeanPostProcessor 的 postProcessAfterInitialization 方法
Bean 初始化完成,可以被使用
容器关闭前,调用 DisposableBean 的 destroy 方法
加载Spring配置文件时,如果Spring配置文件中所定义的Bean类实现了ApplicationContextAware 接口,那么在加载Spring配置文件时,会自动调用ApplicationContextAware 接口中的setApplicationContext,自动的将ApplicationContext注入进来。
在ApplicationContextAware的实现类中,就可以通过这个上下文环境对象得到Spring容器中的Bean。
参考文献:https://zhuanlan.zhihu.com/p/170542769
ApplicationContextAware执行时机:https://www.cnblogs.com/winclpt/articles/7428229.html
一步一步看ApplicationContextAware执行顺序步骤: https://blog.csdn.net/w605283073/article/details/125814360