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

autowire-candidate作什么事情的?

上一篇文章Spring系列第10篇:primary能够解决什么问题?中遇到的问题咱们再来回顾一下,当容器中某种类型的bean存在多个的时候,此时若是咱们从容器中查找这种类型的bean的时候,会报下面这个异常:java

org.springframework.beans.factory.NoUniqueBeanDefinitionException

缘由:当从容器中按照类型查找一个bean对象的时候,容器中却找到了多个匹配的bean,此时spring不知道如何选择了,处于懵逼状态,就会报这个异常。算法

这种异常主要出如今2种场景中:spring

场景1:

从容器容器中查找符合指定类型的bean,对应BeanFactory下面的方法:数据库

<T> T getBean(Class<T> requiredType) throws BeansException;

场景2:

自动注入方式设置为byType的时候,以下:缓存

package com.javacode2018.lesson001.demo8;

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;
    }
}
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       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">

    <bean id="serviceA" class="com.javacode2018.lesson001.demo8.SetterBean$ServiceA"/>
    <bean id="serviceB" class="com.javacode2018.lesson001.demo8.SetterBean$ServiceB"/>

    <bean id="setterBean" class="com.javacode2018.lesson001.demo8.SetterBean" autowire="byType" />
</beans>

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

咱们能够经过primary属性来指定一个主要的bean,当从容器中查找的时候,若是有多个候选的bean符合查找的类型,此时容器将返回primary="true"的bean对象。less

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

<bean id="serviceA" class="com.javacode2018.lesson001.demo8.SetterBean$ServiceA" autowire-candidate="false"/>

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

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       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">

    <bean id="serviceA" class="com.javacode2018.lesson001.demo9.SetterBean$ServiceA" autowire-candidate="false"/>
    <bean id="serviceB" class="com.javacode2018.lesson001.demo9.SetterBean$ServiceB"/>

    <bean id="setterBean" class="com.javacode2018.lesson001.demo9.SetterBean" autowire="byType" />
</beans>

上面咱们将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类中加个方法:

public void setService1(List<IService> service) {//@0
    System.out.println(service); //@1
}

@0:须要注入一个IService的集合,判断一下@1输出的是容器中的serviceA,仍是serviceB,仍是2个都有呢?为何?

案例源码

连接:https://pan.baidu.com/s/1p6rcfKOeWQIVkuhVybzZmQ 
提取码:zr99

Spring系列

  1. Spring系列第1篇:为什么要学spring?

  2. Spring系列第2篇:控制反转(IoC)与依赖注入(DI)

  3. Spring系列第3篇:Spring容器基本使用及原理

  4. Spring系列第4篇:xml中bean定义详解(-)

  5. Spring系列第5篇:建立bean实例这些方式大家都知道?

  6. Spring系列第6篇:玩转bean scope,避免跳坑里!

  7. Spring系列第7篇:依赖注入之手动注入

  8. Spring系列第8篇:自动注入(autowire)详解,高手在于坚持

  9. Spring系列第9篇:depend-on究竟是干什么的?

  10. Spring系列第10篇:primary能够解决什么问题?

更多好文章

  1. Java高并发系列(共34篇)

  2. MySql高手系列(共27篇)

  3. Maven高手系列(共10篇)

  4. Mybatis系列(共12篇)

  5. 聊聊db和缓存一致性常见的实现方式

  6. 接口幂等性这么重要,它是什么?怎么实现?

  7. 泛型,有点难度,会让不少人懵逼,那是由于你没有看这篇文章!

感谢你们的阅读,也欢迎您把这篇文章分享给更多的朋友一块儿阅读!谢谢!

路人甲java

▲长按图片识别二维码关注

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