Spring注入的是接口还是实现类,@Autowired注入对象接口和实现类的区别,同一接口有多个实现类如何注入(转)

问题
为什么xxxController注入的是xxxService,而不是xxxServiceImpl?

controller---->service接口
表面注入的是接口,实际注入的是实现类对象(实现类唯一)。

controller---->serviceImpl实现类
可以对实现类增强,如事务、日志等(AOP动态代理实现)。

结论
@Autowired的对象是通过接口的话,Spring默认会使用jdk动态代理,jdk动态代理只能对实现了接口的类生成代理,而不能针对类,而且还可以对实现类对象做增强得到增强类(增强类与实现类是兄弟关系,增强类不能用实现类接收增强类对象,只能用接口接收)。
————————————————
版权声明:本文为CSDN博主「reppel」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/qq_38691015/article/details/103287839

1、接口:IUserService

public interface IUserService {
    public void sayHello();
}


2、实现类:UserServiceImpl,实现了UserService接口

@Service("UserServiceImpl ")
public class UserServiceImpl implements  IUserService{
 
    // 添加属性:
    private String name;
 
    @Override
    public void sayHello() {
        System.out.println("Hello Spring: " + name);
    }
 
    public String getName() {
        return name;
    }
 
    public void setName(String name) {
        this.name = name;
    }
}


3、业务类:Controller

  

  public class Controller {
        @Autowired
        private IUserService iUserService;
        ......
  }


假如有一个接口 IUserService, UserServiceImpl类实现了接口 UserServiceImpl, 且该接口只有 UserServiceImpl这一个实现类,那么在引用实现类的时候,我们使用的是实现类的接口(像上面程序展示的那样)。Spring会按 byType的方式寻找接口的实现类,将其注入。

假如有另一个实现类 UserServiceImpl2 也实现了接口 IUserService, 这时候再按上面的方式去引用, 在同时存在两个实现类的情况下,会报错。

因为@Autowired 的注入方式是 byType 注入, 当要注入的类型在容器中存在多个时,Spring是不知道要引入哪个实现类的,所以会报错。

可以通过 byName 注入的方式。可以使用 @Resource 或 @Qualifier 注解

@Resource 默认是按照 byName 的方式注入的, 如果通过 byName 的方式匹配不到,再按 byType 的方式去匹配。所以上面的引用可以替换为:

 public class Controller {
        @Resource(name="UserServiceImpl")
        private IUserService iUserService;
        ......
 }


@Qualifier 注解也是 byName的方式,但是与@Resource 有区别,@Qualifier 使用的是 类名。

  public class Controller {
        @Qualifier("UserServiceImpl")
        private IUserService iUserService;
        ......
    }


spring 到底注入接口还是实现类?

<bean id="userService" class="com.lby.demo1.UserServiceImpl">
        <property name="name" value="李四"/>
</bean>


可以看到,在 applicationContext.xml 里bean id为接口名,但是实际的类却是实现类,我们在注入时

// 创建Spring的工厂

ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");


 
// 通过工厂获得类:
// 下面相当于UserService userService = new UserServiceImpl();

UserService userService = (UserService) applicationContext.getBean("userService");


实际上getBean会返回一个父类object,然后我们再强转为子类接口 UserService

在继承关系中,如果一个实例的数据类型是某个子类,那它的数据类型也可以被看做是父类。但是,反过来就不行

接口的实现关系同理,接口相当于接口的实现类

如上面所说,@autowired自动寻找 applicationContext.xml 里 该接口的实现类,而@Resource根据bean id 注入在赋值给注释下的接口

@autowired和@Resourc相当于  UserService userService = (UserService)

applicationContext.getBean("userService");
  • 22
    点赞
  • 55
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值