有时会有这样的需要,在filter中获取spring的bean,这时使用@Autowired注解去自动注入bean就会出现空指针异常,为什么会这样呢?
关于这点,网上的说法大部分是与spring的加载顺序有关,当项目启动时,服务器的加载顺序为:listener->filter->servlet,而spring容器基于servlet,所以在filter中获取不到。然而,在这个说法中还是有些疑问没有被解决,比如说,我写一个类去实现ApplicationContextAware这个接口(后文会介绍),ApplicationContextAware接口的作用大致是:从已有的spring上下文取得已实例化的bean。实现这个接口可以解决filter中获取bean的为空的问题,但这也引出了另一个问题,既然filter加载在servlet之前,那么filter加载时spring容器中应当还没有内容,怎么又能获取到呢?所以我更倾向于另一种解释:filter不归spring管理,所以无法自动注入。这是一个先有鸡还是先有蛋的问题,可能需要对spring加载机制理解的非常透彻才能弄明白吧…
ApplicationContextAware接口
读者可能比较关心filter中注入bean的解决办法,这边提供一种方法。
写一个类实现ApplicationContextAware接口,提供2个静态方法根据类型或name获取bean
当项目启动时,spring能够为我们自动地执行setApplicationContext()方法
注意:使用静态的成员ApplicationContext类型的对象。
@Component
public class SpringUtils implements ApplicationContextAware{
private static ApplicationContext applicationContext;
@Override
public void setApplicationContext(ApplicationContext applicationContext)
throws BeansException {
if(SpringUtils.applicationContext == null){
SpringUtils.applicationContext = applicationContext;
}
}
public static ApplicationContext getApplicationContext(){
return applicationContext;
}
//根据name
public static Object getBean(String name){
return getApplicationContext().getBean(name);
}
//根据类型
public static <T> T getBean(Class<T> clazz){
return getApplicationContext().getBean(clazz);
}
public static <T> T getBean(String name,Class<T> clazz){
return getApplicationContext().getBean(name,clazz);
}
}
我们在filter中就可以直接使用SpringUtils.getBean()去获取bean了
引申一下,如果要获取注册中心上的服务要怎么办?
引用服务使用@Reference注解,显然被引用的对象不在我们当前spring容器中,要引用服务,我们可以在当前容器中创建一个对象去引用服务,调用方法,然后我们在filter中去自动注入这个对象,使用当前容器中的对象执行方法去执行引用的对象的方法。