Spring @Primary作用和实现原理

2,528 阅读3分钟

介绍

今天分享一下Spring中的@Primary注解,Primary的意思是主要的,我们在使用spring的时候,难免会定义多个类型相同的bean,这时候如果不采取一些方法,那么是无法正常使用bean的。

其实在刚开始使用spring的时候,一开始对于某些概念是很不清楚的,所以即使网上去找了很多答案,也是没有真正的理解,所以就导致很多东西都是似懂非懂,就比如根据类型和名称获取bean,一开始这个类型我是有点不太清楚的。

根据类型或者名称获取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的那个。

今天的分享就到这里,感谢你的观看,下期见!