Spring Boot2.x-04Spring Boot基础-使用注解装配bean 中讲了如何将Bean装载到IoC容器中,这里我们说下Bean之间的依赖关系,当然了还是基于注解的方式。
xml的方式去描述Bean之间的依赖关系,请参考以前的博客
Spring-bean之间的关系
Spring-基于注解的配置[02自动装载bean]
举个例子: Manager可以安排Engineer去根据Engineer的类型做不同的工作
接口Engineer的接口方法是coding
package com.artisan.springbootmaster.di.intf;
public interface Engineer {
void coding();
}
假设有个Java程序猿,实现Engineer接口
package com.artisan.springbootmaster.di.intf.impl;
import com.artisan.springbootmaster.di.intf.Engineer;
import org.springframework.stereotype.Service;
@Service
public class JavaEnginerr implements Engineer {
@Override
public void coding() {
System.out.println("Java Engineer works");
}
}
我们在实现类JavaEnginerr 上使用@Service注解,使其成为一个受Spring容器管理的bean。
接下来,我们来看下Manager类
package com.artisan.springbootmaster.di;
import com.artisan.springbootmaster.di.intf.Engineer;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;
@Component
public class Manager {
@Autowired
Engineer engineer;
public void arrange(){
engineer.coding();
}
}
可以通过arrange方法安排engineer工作。 这里Engineer 通过@Autowired让IoC容器自动注入进来。
接着我们使用Java类的方式来初始化IoC容器,通过@Configuration标注其是一个配置类 ,通过ComponetScan来扫描基包下面的标注了注解的类,使其成为受Spring IoC容器托管的bean,方便注入
package com.artisan.springbootmaster.di;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
@Configuration
@ComponentScan(basePackages = "com.artisan.springbootmaster.*")
public class Config {
}
最后,加载Java类的配置,主要是依靠 AnnotationConfigApplicationContext,启动容器获取bean,并调用对应的方法
package com.artisan.springbootmaster.di;
import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
public class DITest {
public static void main(String[] args) {
ApplicationContext applicationContext = new AnnotationConfigApplicationContext(Config.class);
Manager manager = applicationContext.getBean(Manager.class);
manager.arrange();
}
}
运行
23:04:08.018 [main] DEBUG org.springframework.beans.factory.support.DefaultListableBeanFactory - Returning cached instance of singleton bean 'manager'
Java Engineer works
@Autowired会根据属性的类型( by type )找到对应的 Bean 进行注入。
通过结果可以知道,通过注解@Autowired 成功的将JavaEngierr注入到了Manager实例中。
上面这个例子中@Autowired的用法很简单,我们继续来看下@Autowired
当然了,Engineer可能有多个,比如又来了个AndroidEngineer
package com.artisan.springbootmaster.di.intf.impl;
import com.artisan.springbootmaster.di.intf.Engineer;
import org.springframework.stereotype.Service;
@Service
public class AndroidEngineer implements Engineer {
@Override
public void coding() {
System.out.println("Android Engineer works");
}
}
让我们继续运行下DITest,抛出了异常
No qualifying bean of type 'com.artisan.springbootmaster.di.intf.Engineer' available: expected single matching bean but found 2: androidEngineer,javaEnginerr
意思很明显,@Autowired根据类型来匹配Engineer,却发现有2个bean都是Engineer类型 ,这下子Spring不知道注入哪个了。
@Autowired的匹配原则:根据类型找到对应的 Bean,如果对应类型的 Bean 不是唯一 的,那么会继续根据其属性名称和 Bean 的名称进行匹配。如果匹配上,就会使用该 Bean,如果还无法匹配,就会抛出异常。
所以根据上面的原则,比较挫的一个办法(这里只是说可以这么改,但是肯定不推荐这么改)
既然是两个,那我就让bean的名字一样呗
Engineer engineer
保持不变,给这两个Engineer中的任意一个标注@Service(value "engineer")
,指定其bean的名字为engineer,这样根据name就匹配上了,同样不会抛出异常。测试通过。Engineer engineer
改为这两个bean的任意一个名字,@Service标注的实现类Bean的名字为默认第一个字母小写其余保持不变,这样name也能匹配上,同样不会抛出异常。测试通过。这里只是举例验证下Spring @Autowired的匹配规则,实际工作中并不推荐这么改。。。。
@Autowired 是一个默认必须找到对应 Bean 的注解,如果不能确定其标注属性一定会存在并且允许这个被标注的属性为 null , 那么你可以配置@Autowired 属性 required 为 false.
@Autowired既可以标注在属性上,也可以标注在方法上
@Autowired(required = false)
上面通过修改name,使其name保持一致的方式消除了歧义,可以正常的注入,不过并不推荐。
也可以使用@Primary,当然了,也不推荐这么干。 因为另外一个类也可以标注@Primary,Spring又无法知道注入哪个了。
注解@Primary是修改优先权的注解,像上面的两个例子,有2个beanandroidEngineer,javaEnginerr
, 如果我们仅在JavaEnginerr这个类上标注@Primary,意思是告诉Spring IoC 容器 , 当发现有多个同样类型的 Bean ,请优先使用标注了@Primary的这个bean进行注入。
@Qualifier的value属性定义bean的名,该名称将会和@Autowired 组合在一起,通过类型和名称一起找到 Bean。Spring IoC容器中Bean 名称是唯一的标识,通过这个就可以消除歧义了
结果
即使 JavaEnginerr标注了@Primary,但是由于使用了@Qualifier,注入的依然是androidEngineer.
上面的例子,我们是在属性上使用的@Autowired.
如果使构造函数呢?
我们改造下Manager
package com.artisan.springbootmaster.di;
import com.artisan.springbootmaster.di.intf.Engineer;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.stereotype.Component;
@Component
public class Manager {
Engineer engineer;
public void arrange(){
engineer.coding();
}
public Manager(@Autowired @Qualifier("javaEnginerr") Engineer engineer){
this.engineer = engineer;
}
}
运行