Spring高级装配之处理自动装配的歧义性

通常bean的自动装配能够给我们提供很大的方便,它会减少装配应用程序组件时所需要的显示配置的麻烦。不过,仅有一个bean匹配所需的结果时,自动装配才是有效的。如果符合条件的bean不只一个,这时就会阻碍Spring自动装配属性、构造器参数或方法参数。
下面我们就来制造一种自动装配歧义性的情况:

public interface Dessert {

    String sayHello();

}

@Service
public class Cake implements Dessert{

    @Override
    public String sayHello() {
        return "Hello World!";
    }

}
@Service
public class IceCream implements Dessert {

    @Override
    public String sayHello() {
        return "Hello World!";
    }

}
@Service
public class Cookies implements Dessert{

    @Override
    public String sayHello() {
        return "Hello World!";
    }

}
@Autowired
private Dessert dessert;
    
public void sayHello(){
    System.out.println(dessert.sayHello());
}

在Spring初始化bean之后,他会尽可能得去满足bean的依赖,在本例中,依赖是通过带有@Autowired注解的方法进行声明的。通常为了当要注入的bean不存在时,不至于抛异常,将@Autowired的required属性设置为false。
在进行组件扫描的时候,能够发现它们并将其创建为Spring应用上下文里面的bean,但现在Spring要注入的Dessert类型有三个实现,所有Spring就不能决定要注入那一个了,所以就抛了异常:

nested excpetion is
    org.springframework.beans.factory.NoUniqueBeanDefinitionException:
No quanlifying bean of type [com.ambiguity.service.Dessert] is defined:
    expeted single matching bean but found 3: cake,cookies,iceCream

但是不要担心,Spring提供了很多解决这个问题的方案,你可以将可选bean中的一个设置为首选(primary)的bean,或者使用限定符(qualifier)来帮助Spring将可选的bean的范围缩小到只有一个bean。

1.标示首选的bean

在Spring中可以通过@Primary注解将一个bean标记为首选bean,比如下面我们将IceCream bean声明为首选bean:

@Service
@Primary
public class IceCream implements Dessert {

    @Override
    public String sayHello() {
        return "Hello World!";
    }
}

或者,如果你通过bean配置显示地声明IceCream,那么@Bean方法应该如下所示:

@Configuration
public class IceCreamBean {

    @Bean
    @Primary
    public Dessert iceCream(){
        return new IceCream();
    }
}

如果你使用的是XML装配Bean,同样可以使用这样的功能:

<bean id="iceCream" class="com.ambiguity.service.impl.IceCream" primary="true" />

不管你使用哪种方式来标记首选bean,效果都是一样的,都是告诉Spring在遇到歧义性的时候要选择首选Bean。
但是当两个bean都被标记了首选bean,那么Spring就有无法正常工作了。这时就要使用Spring提供的另一个强大的机制了,那就是限定符。

2.限定自动装配的bean

Spring的限定符能够在所有可选的bean上进行缩小范围的操作,最终能够达到只有一个bean满足所有规定的限制条件。如果将所有限定符都用上后依然存在歧义性,那么你可以继续使用限定符来缩小选择范围。
@Qualifier注解是使用限定符的重要方式。它可以与@Autowired和@Inject协同使用,在注入的时候指定想要注入进去的是哪一个bean。例如,我们确保将IceCream注入到setDessert()中:

@Service
public class AmbiguityTest {

    @Autowired(required=false)
    @Qualifier("iceCream")
    private Dessert dessert;
    
    
    public void sayHello(){
        System.out.println(dessert.sayHello());
    }
}

为@Qualifier注解所设置的参数就是想注入的bean的ID。所有使用了@Component注解声明的的类都会创建为bean,并且bean的ID默认为类名首字母小写,即:iceCream。因此,@Qualifier("iceCream")指向的是组件扫描时所创建的bean。并且这个bean是IceCream的实例。
更准确的讲,@Qualifier("iceCream")所引用的bean具有String类型的"iceCream"作为限定符。默认情况下bean的限定符与bean的ID相同。
基于默认的bean ID作为限定符是非常简单的,但这有可能会引发一些问题。如果你重构了IceCream,更改了了这个类的名,那么我们使用@Qualifier("iceCream")注入时就会发生异常,不要担心我们可以自定义bean的限定符。

创建自定义限定符

我们可以为bean设置自己的限定符,我们只需在bean声明上添加@Qualifier注解:

@Service
@Primary
@Qualifier("cold")
public class IceCream implements Dessert {

    @Override
    public String sayHello() {
        return "Hello World!";
    }

}

在使用bean的时候只需要在setDessert()方法上加上@Qualifier("cold"):

@Service
public class AmbiguityTest {

    @Autowired(required=false)
    @Qualifier("cold")
    private Dessert dessert;
    
    
    public void sayHello(){
        System.out.println(dessert.sayHello());
    }
}

如果bean是通过@Bean的方式显示的装配的,@Qualifier也可以和@Bean注解一起使用:

@Configuration
public class IceCreamBean {

    @Bean
    @Qualifier("cold")
    public Dessert iceCream(){
        return new IceCream();
    }
}

使用自定义的限定符注解

当两个bean的限定符相同时,在我们就再次遇到了歧义性:

@Service
@Qualifier("cold")
public class Cookies implements Dessert{

    @Override
    public String sayHello() {
        return "Hello World!";
    }

}

我们也许会想到使用多个@Qualifier注解,像这样:

@Service
@Qualifier("cold")
@Qualifier("creamy")
public class IceCream implements Dessert {

    @Override
    public String sayHello() {
        return "Hello World!";
    }

}

但事实上,Java不予许同一个条目上重复出现相同类型的多个注解。这时候我们可以通过自定义注解的方式解决这一问题:

@Target({ElementType.CONSTRUCTOR, ElementType.FIELD, ElementType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Qualifier
public @interface Cold {

}

@Target({ElementType.CONSTRUCTOR, ElementType.FIELD, ElementType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Qualifier
public @interface Creamy {

}

这时候如果我们不想用@Qualifier注解的时候,就可以使用我们自定义的限定符注解@Cold和@Creamy:

@Service
@Cold
@Creamy
public class IceCream implements Dessert {

    @Override
    public String sayHello() {
        return "Hello World!";
    }

}

同样,也可以自定义注解@Fruity注解,供Cookies bean使用:

@Service
@Cold
@Fruity
public class Cookies implements Dessert{

    @Override
    public String sayHello() {
        return "Hello World!";
    }
}

这样解决自动自动装配歧义性的解决方案就全部介绍完了!下节我们将会介绍如何在不同作用域中声明bean。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,569评论 4 363
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,499评论 1 294
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,271评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,087评论 0 209
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,474评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,670评论 1 222
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,911评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,636评论 0 202
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,397评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,607评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,093评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,418评论 2 254
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,074评论 3 237
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,092评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,865评论 0 196
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,726评论 2 276
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,627评论 2 270

推荐阅读更多精彩内容

  • 本章内容: Spring profile 条件化的bean声明 自动装配与歧义性 bean的作用域 Spring表...
    谢随安阅读 1,147评论 0 5
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,100评论 18 139
  • Spring Boot 参考指南 介绍 转载自:https://www.gitbook.com/book/qbgb...
    毛宇鹏阅读 46,365评论 6 343
  • 2017.09.11 周一 晴 今天是儿子吃小饭桌的第一天。幼儿园三年,小学三年都没有让他吃过小饭桌。今年因为特殊...
    戴骁勇阅读 580评论 0 0
  • 电梯里不敢看你尴尬的脸,只是低着头摆弄着手机,胸中始终还是堵着一口气,什么也不说彼此沉默着,气氛僵持而冷漠。不...
    若即若离you阅读 318评论 0 0