Spring @Primary作用和实现原理详解
作者:刘牌
根据类型或者名称获取bean
根据名称获取bean这是很好理解的,在spring中,定义bean的方式有很多,不同通常都是@Component注解和@Bean注解来注册bean,使用@Component的时候我们可以指定名称,如果不指定,则就为类名的名称,只不过首字母小写,使用@Bean的时候,也可以指定,不指定的话则为方法名称,但是根据类型获取,之前不是很理解,其实类型就是接口,我们知道接口一般是不做任何实现的,由实现类来进行具体逻辑实现,可以有多个实现类,比如UserService接口有UserServiceImpl1和UserServiceImpl2两个实现类,那么我们就可以说UserServiceImpl1和UserServiceImpl2的类型是UserService,在spring中,这两个实现类如果定义成bean,那么都会被注册进spring IOC容器中,但是在获取的时候是有问题的。
byName
如果根据名称获取,那么是没问题的,因为注册进IOC容器时是注册了两个不同名称的bean,所以根据名称获取,只获取到了一个bean,所以能获取成功。
byType
如果根据类型获取,因为注册了两个bean,但是它们的类型是相同的,这时候直接根据类型获取,那么就会获取到两个bean,spring不能返回那个,所以就抛出异常。
No qualifying bean of type 'io.steakliu.spring.ioc.annotation.primary.PrimaryService' available: expected single matching bean but found 2: primaryServiceImpl1,primaryServiceImpl2
解决方案
存在多个类型的bean时,可以通过@Qualifier
注解指定名称,我们也可以在注入bean的时候指定具体的bean名称,不过这样的话就会变得很变扭,所以使用@Qualifier
还是比较合理和规范,当然,也可以使用@Primary来,但实际上@Primary能解决的问题我觉得只是给了一个默认值,这样在获取bean的时候不会报错,当需要注入同一类型下的其他bean时,依旧需要指定名称。
实现原理
如果我们使用@Component或@Bean注解注册bean的时候,spring启动后会扫描工程路径下的包,然后找出相应的类,并解析相应的Bean,如果带有@Primary注解,那么就会将BeanDefinition的primary属性设置为true,然后将BeanDefinition注册到BeanFactory,如下所示。
在获取bean的时候,首先会获取类型下面的所有bean,然后再根据bean的名字去获取BeanDefination,最后选择primary属性为true的bean返回,如果同一类型下的bean都加了@Primary注解,那么获取的时候会报错。
以上就简单的分析了@Primary的源码,其实是比较简单的,primary作为bean定义信息中的一个属性,spring扫描到了标有这个注解,就会将BeanDefinition的primary设置为true,获取获取bean的时候,就再次取出Beandefinition,判断primary属性,然后从获取的候选bean中选出primary为true的那个。
到此这篇关于Spring @Primary注解作用和实现原理详解的文章就介绍到这了,更多相关Spring @Primary作用内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!