spring使用@Autowire存在两个同类型bean的时候如何注入以及在spring中bean的名称与什么有关

先上结论:

当容器中的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在不同情况下与类名或者函数名的关系。