1. 来源:
@Resource 出自 JDK自带工具包, 路径:
javax.annotation.Resource
@Autowired 出自spring-beans(springboot项目的spring-boot-starter下), 路径:
org.springframework.beans.factory.annotation.Autowired
2. 举例:
2-1: 一个 service接口, 两个service接口实现类
public interface TestService {
}
@Service
public class TestServiceImpl implements TestService {
}
@Service
public class Test2ServiceImpl implements TestService {
}
2-2: 在控制层中引用对象
@RestController
public class TestController {
@Resource(name = "testServiceImpl")
private TestService testService;
@Autowired
@Qualifier("test2ServiceImpl")
private TestService testService2;
}
3. 相同: 两个注解的共同作用"实例化对象"
4. 不同:
4-1: @Resource:
(1) 不用 name 注解属性:
默认先使用 private(修饰符) TestService(类) testService(变量名); 中的 testService , 去spring 容器中 找到 名称 == testService 的实现类, 然后进行对象实例化.
如果 通过名称 找不到的时候, 会用 TestService 类型去spring 容器中 找到 类 = TestService的 实现类, 然后进行对象实例化.
注: 以上 示例中, 有 1个接口, 以及2个该接口的实现类.
错误写法: private TestService testService;
通过 testService 名称去查找的话, 程序会报错, 因为程序此时发现有 2个该接口的实现类, 不知道具体该实例化哪一个.
正确写法: private TestService testServiceImpl;
通过变量名(testServiceImpl)指定需要实例化的实现类.
如果 1个接口, 仅有 1个实现类, 则 private TestService testService; 不存在以上问题
(2) 用 name 注解属性:
指定 name ="testServiceImpl" 后, 去spring 容器中 找到 名称 == testServiceImpl的实现类, 如果通过名称 找不到的时候, 程序直接报错, 不会再自动 按照 类型 去查找
4-2: @Autowired
(1) private TestService testService2; 上使用该注解
表明 先用 TestService 类型 去spring 容器中 找到 类 == TestService的 实现类, 然后进行对象实例化.
如果 通过 类 找不到的时候, 会用 testService2 名称去spring 容器中 找到 名称 == testService2 的实现类, 然后进行对象实例化.
(2) 该注解 和 @Qualifier("test2ServiceImpl") 一起使用时, 则 等同于 @Resource(name="test2ServiceImpl") 注解, 即-查找 名称 == test2ServiceImpl 的类, 找不到则报错
注: 接口是无法被实例化的, 只有接口的实现类可以. 所以在控制层, 用业务层接口, 实际上查找的是 业务层接口对应的具体某一个实例类(如果一个接口有多个实现类, 则一定要指明具体生成哪一个实例类)