背景:
在使用谷歌开源的本地缓存解决经常查询数据库导致的查询效率低下,将从数据库查询好的数据放入到缓存中,然后设计过期时间,接着设计一个get方法缓存汇总获取数据,进一步将整个流程封装成一个CacheSerice,然后在Controller层调用这个Service,从Service中获取数据。
问题:
需要对CacheService进行初始化,设计的初衷是:当Service的bean被加载之后,其中的缓存数据就已经被初始化(即利用数据库查询Service获取数据,并塞入缓存),而这个初始化的过程被我放到了CacheService类的构造函数中。结果在发布的时候就一直报空指针。
@Service("test")
public class Test implements IAppnameCache {
@Autowired
IAppnameService iAppnameService;
public Test(){
iAppnameService.queryAppname();// 抛出空指针
}
@Override
public List get(String app){
return iAppnameService.queryAppname();
}
}
问题定位:
经过查询日志,发现是CacheService的构造函数在执行的时候发生空指针问题。那么有可能是引入的谷歌开源库的问题有可能不是,采用排除法很快就发现了不是这个库的问题,不含谷歌开源库的测试类采用这种写法也发生了空指针的问题。
问题思考:
既然跟引入的谷歌开源库没有关系,那就说明当CacheService被构造的时候(采用构造函数),里面依赖的其他bean还没有被构造出来,因而导致空指针问题。针对这个问题进一步对Spring的bean构造过程进行研究。
Spring的bean加载过程:
bean的主要生成过程如下:
1,AbstractBeanFactory.getBean(String)
2,AbstractBeanFactory.doGetBean(String, Class, Object[], boolean)
3,DefaultSingletonBeanRegistry.getSingleton(String)
4,AbstractAutowireCapableBeanFactory.createBean(String, RootBeanDefinition, Object[])
5,AbstractAutowireCapableBeanFactory.doCreateBean(String, RootBeanDefinition, Object[])
6,AbstractAutowireCapableBeanFactory.createBeanInstance(String, Root