最近做项目,无意间用到了spring注入集合的问题,具体问题如下:
在spring配置文件中配置map如下:
image.png
在项目中使用map如下:
@Autowired
private Map<String, HandlerInterceptor> handlerMap;
奇怪的是,当我使用handlerMap.get("async")获取处理类时,获取不到具体的bean,这是怎么回事呢?
问题可能出现的原因:
1、map的key值被覆盖,xml配置不起作用
2、注入的map和xml配置的map不是同一个bean
问题排查:
首先我更改了xml配置,把bean的id修改成了handlerMap1重启服务,发现注入的handlerMap值还是存在的,并且map的key值是对应value bean的id。这样基本上排除了提出的第一个假设。那么,我们来看下第二种假设。
通过打断点debug等手段,我们很容易的得到结果,注入的map和xml配置的map实例化的bean不是同一个,问题解决。
那么,为什么我xml配置好一个map对象,注入的时候又会给我创建一个对象呢?
我们来看下@Autowired的注入方式:按照类型注入
然后,我们看下Autowired的注入解析流程,在实例化bean的时候,发现
1555401671(1).png
org.springframework.beans.factory.support.DefaultListableBeanFactory#doResolveDependency
这个是4.0的代码,5.0的代码在org.springframework.beans.factory.support.DefaultListableBeanFactory#resolveMultipleBeans这个方法中。
根据上面的截图我们会发现,当我们用Autowired注入一个map或者其他集合类型时,spring不是根据beanName取bean然后赋值,而是根据设置的type从容器中一次性取出所有符合的bean直接放入到集合中。
现在知道问题的根源了,那我们怎么解决这个问题呢?
既然Autowired是按照类型注入的,那我们是不是可以选择按照名称注入呢@Resource
经测试 @Resource完全能解决问题!
网友评论