Spring系列第11篇:bean中的autowire-candidate又是干什么的?

setterBean的autowire设置的是byType,即按setter方法的参数类型自动注入,SetterBean的setService的类型是IService,而IService类有2个实现类:ServiceA和ServiceB,而容器容器中刚好有这2个实现类的bean:serviceA和serviceB,所以上面代码会报错,不知道注入的时候选择那个对象注入。

我们可以通过primary属性来指定一个主要的bean, 《一线大厂Java面试题解析+后端开发学习笔记+最新架构讲解视频+实战项目源码讲义》无偿开源 威信搜索公众号【编程进阶路】 当从容器中查找的时候,如果有多个候选的bean符合查找的类型,此时容器将返回primary="true"的bean对象。

spring还有一种方法也可以解决这个问题,可以设置某个bean是否在自动注入的时候是否为作为候选bean,通过bean元素的autowire-candidate属性类配置,如下:

autowire-candidate:设置当前bean在被其他对象作为自动注入对象的时候,是否作为候选bean,默认值是true。

来举例说明一下,以上面的setter注入的案例先来说一下注入的过程:

容器在创建setterBean的时候,发现其autowire为byType,即按类型自动注入,此时会在SetterBean类中查找所有setter方法列表,其中就包含了setService方法,setService方法参数类型是IService,然后就会去容器中按照IService类型查找所有符合条件的bean列表,此时容器中会返回满足IService这种类型并且autowire-candidate="true"的bean,刚才有说过bean元素的autowire-candidate的默认值是true,所以容器中符合条件的候选bean有2个:serviceA和serviceB,setService方法只需要一个满足条件的bean,此时会再去看这个列表中是否只有一个主要的bean(即bean元素的primary=“ture”的bean),而bean元素的primary默认值都是false,所以没有primary为true的bean,此时spring容器懵了,不知道选哪个了,此时就报错了,抛出NoUniqueBeanDefinitionException异常

从上面过程中可以看出将某个候选bean的primary置为true就可以解决问题了。

或者只保留一个bean的autowire-candidate为true,将其余的满足条件的bean的autowire-candidate置为false,此时也可以解决这个问题,下面我们使用autowire-candidate来解决上面问题看一下效果:

SetterBean.java

package com.javacode2018.lesson001.demo9;

public class SetterBean {

public interface IService {} //@1

public static class ServiceA implements IService {} //@2

public static class ServiceB implements IService {} //@3

private IService service;

public void setService(IService service) {

this.service = service;

}

@Override

public String toString() {

return “SetterBean{” +

“service=” + service +

‘}’;

}

}

autowireCandidateBean.xml

xmlns:xsi=“http://www.w3.org/2001/XMLSchema-instance”

xsi:schemaLocation="http://www.springframework.org/schema/beans

http://www.springframework.org/schema/beans/spring-beans-4.3.xsd">

上面我们将serviceA的autowire-candidate置为false了,serviceA在被其他bean自动按照类型注入的时候,将不再放入候选名单中

测试用例

package com.javacode2018.lesson001.demo9;

import com.javacode2018.lesson001.demo5.IocUtils;

import com.javacode2018.lesson001.demo8.NormalBean;

import com.javacode2018.lesson001.demo8.PrimaryBean;

import org.junit.Test;

import org.springframework.context.support.ClassPathXmlApplicationContext;

import java.util.Arrays;

import java.util.Map;

/**

* 公众号:路人甲Java,工作10年的前阿里P7分享Java、算法、数据库方面的技术干货!坚信用技术改变命运,让家人过上更体面的生活!

* bean元素的autowire-candidate可以设置当前bean是否作为其他bean自动注入的候选bean

*/

public class AutowireCandidateTest {

@Test

public void setterBean() {

String beanXml = “classpath:/com/javacode2018/lesson001/demo9/autowireCandidateBean.xml”;

ClassPathXmlApplicationContext context = IocUtils.context(beanXml);

System.out.println(context.getBean(SetterBean.class)); //@1

SetterBean.IService service = context.getBean(SetterBean.IService.class); //@2

System.out.println(service);

}

}

@1:查找容器中SetterBean类型的bean对象

@2:查找容器中SetterBean.IService接口类型的bean,实际上面容器中serviceA和serviceB都是这种类型的

下面我们运行一下,看看输出:

SetterBean{service=com.javacode2018.lesson001.demo9.SetterBean$ServiceB@29176cc1}

com.javacode2018.lesson001.demo9.SetterBean$ServiceB@29176cc1

注意一下输出,2行输出中都是ServiceB,因为serviceB的autowire-candidate是默认值true,自动注入的时候作为候选bean,而serviceA的autowire-candidate是false,自动注入的时候不作为候选bean,所以上面输出的都是serviceB。

autowire-candidates属性解析源码

=========================

beans元素是xml中定义bean的根元素,beans元素有个default-autowire-candidates属性,用于定义哪些bean可以作为候选者,default-autowire-candidates的值是个通配符如:

default-autowire-candidates=“*Service”

再来说一下bean元素的autowire-candidate属性,这个属性有3个可选值:

  • default:这个是默认值,autowire-candidate如果不设置,其值就是default

  • true:作为候选者

  • false:不作为候选者

spring中由beans元素的default-autowire-candidates和bean元素的autowire-candidate来决定最终bean元素autowire-candidate的值,我们来看一下bean元素autowire-candidates的解析源码:

源码位置:org.springframework.beans.factory.xml.BeanDefinitionParserDelegate#parseBeanDefinitionAttributes

主要代码如下:

//获取bean元素的autowire-candidate元素,autowire-candidate如果不设置,其值就是default

String autowireCandidate = ele.getAttribute(AUTOWIRE_CANDIDATE_ATTRIBUTE);

//判断bean元素的autowire-candidate元素是否等于"default"或者是否等于""

if (isDefaultValue(autowireCandidate)) {

//获取beans元素default-autowire-candidates属性值

String candidatePattern = this.defaults.getAutowireCandidates();

//判断获取beans元素default-autowire-candidates属性值是否为空,default-autowire-candidates默认值就是null

if (candidatePattern != null) {

//判断bean的名称是否和default-autowire-candidates的值匹配,如果匹配就将bean的autowireCandidate置为true,否则置为false

String[] patterns = StringUtils.commaDelimitedListToStringArray(candidatePattern);

bd.setAutowireCandidate(PatternMatchUtils.simpleMatch(patterns, beanName));

}

}else {

//判断bean的autowire-candidate的值是否等于"true"

bd.setAutowireCandidate(TRUE_VALUE.equals(autowireCandidate));

}

如果上面判断都没有进去,autowireCandidate属性默认值就是true,这个在下面定义的:

org.springframework.beans.factory.support.AbstractBeanDefinition#autowireCandidate

private boolean autowireCandidate = true;

所有的bean元素最后都会被解析为spring中的org.springframework.beans.factory.config.BeanDefinition对象,关于BeanDefinition以后我们会细说

问题

==

对上面的案例做个扩展,SetterBean类中加个方法:

你可能感兴趣的:(Java,经验分享,架构,java)