Spring中接口上使用@Autowired为何注入的是实现类而非接口?

2026-05-25 21:271阅读0评论SEO资源
  • 内容介绍
  • 文章标签
  • 相关推荐

本文共计1006个文字,预计阅读时间需要5分钟。

Spring中接口上使用@Autowired为何注入的是实现类而非接口?

在Spring框架中,使用`@Autowired`注解可以自动注入依赖。以下是对您提供的内容的简化

目录:`@Autowired`注入到接口上,实际获取的是实现类;为什么它会拿到`userServiceImpl`?一个接口有多个实现类时,`@Autowired`如何选择?

`@Autowired`注解用于接口上时,会自动注入实现类;为何会找到`userServiceImpl`?如果接口有多个实现类,`@Autowired`如何决定注入哪个实现类?

目录
  • @Autowired加到接口上但获取的是实现类
    • 问题
    • 为什么他会拿到userServiceImpl?
  • @Autowired一个接口有多个实现类
    • 例如

@Autowired加到接口上但获取的是实现类

问题

Spring的@Autowired加到接口上但获取的是实现类?

    /* 类 @Controller注解,会在spring容器中实例化对象 */     @Controller     public class UserContoller{         @Autowired        // 先按类型找,然后按id为属性名去找         private UserService userService;         //为什么他会拿到userServiceImpl?         // @Autowired会帮你按UserService的类型去容器中找唯一bean对象         // 1、容器没有该类型的对象:报错         // 2、容器中有该类型的唯一bean对象,就将该唯一bean对象赋值给该属性         ///3、容器中有多个该类型的唯一bean对象,         //     它会再根据该属性名去容器中找,         //     看看容器中的哪个bean对象的id值和该属性名一致,         //     如果有,就将容器中该对象赋值给该属性,如果没有报错。     }         /* 接口  */     public interface UserService{}          /* 类  @Service注解,会在spring容器中实例化对象 */     @Service     public class UserServiceImpl implements UserService{}

为什么他会拿到userServiceImpl?

@Autowired先按类型找,然后再按id为属性名去找

他会帮你按UserService的类型去容器中找唯一bean对象

  • 1.容器没有该类型的对象:报错
  • 2.容器中有该类型的唯一bean对象,就将该唯一bean对象赋值给该属性
  • 3.容器中有多个该类型的唯一bean对象,

它会再根据该属性名去容器中找,看看容器中的哪个bean对象的id值和该属性名一致,如果有,就将容器中该对象赋值给该属性,如果没有报错。

然后通过多态的向上转型就赋值成功。等价于之前手动赋值

UserService userService = new UserServiceImpl();

@Autowired一个接口有多个实现类

@Autowired是spring的注解,默认使用的是byType的方式向Bean里面注入相应的Bean。

例如

@Autowired private UserService userService;

这段代码会在初始化的时候,在spring容器中寻找一个类型为UserService的bean实体注入,关联到userService的引入上。

但是如果UserService这个接口存在多个实现类的时候,就会在spring注入的时候报错,具体如下:

public class UserService1 implements UserService public class UserService2 implements UserService

当存多个UserService的实现类时,错误信息如下:

2016-08-05 14:53:53,795 ERROR [org.springframework.test.context.TestContextManager] - <Caught exception while allowing TestExecutionListener [org.springframework.test.context.support.DependencyInjectionTestExecutionListener@14a2f921] to prepare test instance [UserServiceTest@3c87521]>
org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'UserServiceTest': Injection of autowired dependencies failed; nested exception is org.springframework.beans.factory.BeanCreationException: Could not autowire field: private yjc.demo.service.UserService UserServiceTest.userService; nested exception is he[yjc.demo.service.UserService] is defined: expected single matching bean but found 2: userService1,userService2

抛出了org.springframework.beans.factory.BeanCreationException,而原因是注入的时候发现有2个匹配的bean,但是不知道要注入哪一个:expected single matching bean but found 2: userService1,userService2

那么如何应对多个实现类的场景呢,看一下代码:

@Autowired private UserService userService1;   @Autowired private UserService userService2;   @Autowired @Qualifier(value = "userService2") private UserService userService3;   @Test public void test(){          System.out.println(userService1.getClass().toString());          System.out.println(userService2.getClass().toString());          System.out.println(userService3.getClass().toString()); }

运行结果:

class yjc.demo.serviceImpl.UserService1
class yjc.demo.serviceImpl.UserService2
class yjc.demo.serviceImpl.UserService2

运行结果成功,说明了2种处理多个实现类的方法:

1.变量名用userService1,userService2,而不是userService。

通常情况下@Autowired是通过byType的方法注入的,可是在多个实现类的时候,byType的方式不再是唯一,而需要通过byName的方式来注入,而这个name默认就是根据变量名来的。

2.通过@Qualifier注解来指明使用哪一个实现类,实际上也是通过byName的方式实现。

Spring中接口上使用@Autowired为何注入的是实现类而非接口?

由此看来,@Autowired注解到底使用byType还是byName,其实是存在一定策略的,也就是有优先级。优先用byType,而后是byName。

以上为个人经验,希望能给大家一个参考,也希望大家多多支持自由互联。

本文共计1006个文字,预计阅读时间需要5分钟。

Spring中接口上使用@Autowired为何注入的是实现类而非接口?

在Spring框架中,使用`@Autowired`注解可以自动注入依赖。以下是对您提供的内容的简化

目录:`@Autowired`注入到接口上,实际获取的是实现类;为什么它会拿到`userServiceImpl`?一个接口有多个实现类时,`@Autowired`如何选择?

`@Autowired`注解用于接口上时,会自动注入实现类;为何会找到`userServiceImpl`?如果接口有多个实现类,`@Autowired`如何决定注入哪个实现类?

目录
  • @Autowired加到接口上但获取的是实现类
    • 问题
    • 为什么他会拿到userServiceImpl?
  • @Autowired一个接口有多个实现类
    • 例如

@Autowired加到接口上但获取的是实现类

问题

Spring的@Autowired加到接口上但获取的是实现类?

    /* 类 @Controller注解,会在spring容器中实例化对象 */     @Controller     public class UserContoller{         @Autowired        // 先按类型找,然后按id为属性名去找         private UserService userService;         //为什么他会拿到userServiceImpl?         // @Autowired会帮你按UserService的类型去容器中找唯一bean对象         // 1、容器没有该类型的对象:报错         // 2、容器中有该类型的唯一bean对象,就将该唯一bean对象赋值给该属性         ///3、容器中有多个该类型的唯一bean对象,         //     它会再根据该属性名去容器中找,         //     看看容器中的哪个bean对象的id值和该属性名一致,         //     如果有,就将容器中该对象赋值给该属性,如果没有报错。     }         /* 接口  */     public interface UserService{}          /* 类  @Service注解,会在spring容器中实例化对象 */     @Service     public class UserServiceImpl implements UserService{}

为什么他会拿到userServiceImpl?

@Autowired先按类型找,然后再按id为属性名去找

他会帮你按UserService的类型去容器中找唯一bean对象

  • 1.容器没有该类型的对象:报错
  • 2.容器中有该类型的唯一bean对象,就将该唯一bean对象赋值给该属性
  • 3.容器中有多个该类型的唯一bean对象,

它会再根据该属性名去容器中找,看看容器中的哪个bean对象的id值和该属性名一致,如果有,就将容器中该对象赋值给该属性,如果没有报错。

然后通过多态的向上转型就赋值成功。等价于之前手动赋值

UserService userService = new UserServiceImpl();

@Autowired一个接口有多个实现类

@Autowired是spring的注解,默认使用的是byType的方式向Bean里面注入相应的Bean。

例如

@Autowired private UserService userService;

这段代码会在初始化的时候,在spring容器中寻找一个类型为UserService的bean实体注入,关联到userService的引入上。

但是如果UserService这个接口存在多个实现类的时候,就会在spring注入的时候报错,具体如下:

public class UserService1 implements UserService public class UserService2 implements UserService

当存多个UserService的实现类时,错误信息如下:

2016-08-05 14:53:53,795 ERROR [org.springframework.test.context.TestContextManager] - <Caught exception while allowing TestExecutionListener [org.springframework.test.context.support.DependencyInjectionTestExecutionListener@14a2f921] to prepare test instance [UserServiceTest@3c87521]>
org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'UserServiceTest': Injection of autowired dependencies failed; nested exception is org.springframework.beans.factory.BeanCreationException: Could not autowire field: private yjc.demo.service.UserService UserServiceTest.userService; nested exception is he[yjc.demo.service.UserService] is defined: expected single matching bean but found 2: userService1,userService2

抛出了org.springframework.beans.factory.BeanCreationException,而原因是注入的时候发现有2个匹配的bean,但是不知道要注入哪一个:expected single matching bean but found 2: userService1,userService2

那么如何应对多个实现类的场景呢,看一下代码:

@Autowired private UserService userService1;   @Autowired private UserService userService2;   @Autowired @Qualifier(value = "userService2") private UserService userService3;   @Test public void test(){          System.out.println(userService1.getClass().toString());          System.out.println(userService2.getClass().toString());          System.out.println(userService3.getClass().toString()); }

运行结果:

class yjc.demo.serviceImpl.UserService1
class yjc.demo.serviceImpl.UserService2
class yjc.demo.serviceImpl.UserService2

运行结果成功,说明了2种处理多个实现类的方法:

1.变量名用userService1,userService2,而不是userService。

通常情况下@Autowired是通过byType的方法注入的,可是在多个实现类的时候,byType的方式不再是唯一,而需要通过byName的方式来注入,而这个name默认就是根据变量名来的。

2.通过@Qualifier注解来指明使用哪一个实现类,实际上也是通过byName的方式实现。

Spring中接口上使用@Autowired为何注入的是实现类而非接口?

由此看来,@Autowired注解到底使用byType还是byName,其实是存在一定策略的,也就是有优先级。优先用byType,而后是byName。

以上为个人经验,希望能给大家一个参考,也希望大家多多支持自由互联。