先上结论:
当容器中的bean为单例时,@Autowire为ByType的方式注入,被注入的成员的名称可以任意取名。
当容器中的bean存在多个的情况下,@Autowire为ByName的方式注入,ByName是将bean单例池中的key(bean的名字)与被注入的成员变量的名称匹配,而不是与被注入的成员变量的类型匹配。
关于单例池中的key(bean的name)与什么有关:
如果在类上面加上bean化注解(@service、@controller、@component、@repository),则此bean的name就是这个类首字母小写的驼峰命名。
举个栗子:
@Componet
public class TestObject{
}
则TestObject类在容器中的bean的name就是testObject
如果在方法上加@Bean注解,则bean的name就是 方法的名字
再举个栗子:
@Componet
public class TestObject{
@Bean
public TestObject getTestObjectBean(){
return new TestObject();
}
}
则此时spring容器中会有两个TestObject类型的bean,一个叫testObject,一个叫getTestObjectBean。
注意:容器中同时有两个相同name的bean,spring启动会报错!
接下来验证一下容器中同事有两个不同类型的bean的时候,@Autowire如何注入的问题,上一下之前写过的验证代码:
@Component
public class UserService implements UserServiceInterface {
@Bean
public UserService getUserService() {
return new UserService();
}
}
此时容器中有两个 UserService类型的bean,一个叫userService,一个叫getUserService
@Component
public class Config {
@Autowired
private UserServiceInterface getUserService;
@Autowired
private UserServiceInterface userService;
public UserServiceInterface getUserService() {
return this.userService;
}
public UserServiceInterface getGetUserService() {
return this.getUserService;
}
}
在Config里面注入这两个bean,如果被注入的对象这个时候取名非userService和getUserService,则在spring启动的时候会发生没有匹配bean的错误,如下:
接下来是拿到spring上下文验证的代码段:
public class Test {
public static void main(String[] args) {
AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class);
Config config = context.getBean("config", Config.class);
UserServiceInterface userService = context.getBean("userService", UserServiceInterface.class);
UserServiceInterface getUserService = context.getBean("getUserService", UserServiceInterface.class);
System.out.printf("config.getUserService() == userService? ");
System.out.println(config.getUserService() == userService);
System.out.printf("config.getGetUserService() == getUserService ");
System.out.println(config.getGetUserService() == getUserService);
}
}
运行一下:
控制台倒数3-6行打印的结果也印证了结论中的bean的name在不同情况下与类名或者函数名的关系。