@Component
:通用的注解,可标注任意类为 Spring
组件。如果一个 Bean 不知道属于哪个层,可以使用@Component
注解标注。
@Repository
: 对应持久层即 Dao 层,主要用于数据库相关操作。
@Service
: 对应服务层,主要涉及一些复杂的逻辑,需要用到 Dao 层。
@Controller
: 对应 Spring MVC 控制层,主要用户接受用户请求并调用 Service 层返回数据给前端页面。
@Component
注解作用于类,而@Bean
注解作用于方法。
@Component
通常是通过类路径扫描来自动侦测以及自动装配到 Spring 容器中(我们可以使用 @ComponentScan
注解定义要扫描的路径从中找出标识了需要装配的类自动装配到 Spring 的 bean 容器中)。@Bean
注解通常是我们在标有该注解的方法中定义产生这个 bean,@Bean
告诉了 Spring 这是某个类的实例,当我需要用它的时候还给我。
@Bean
注解比 @Component
注解的自定义性更强,而且很多地方我们只能通过 @Bean
注解来注册 bean。比如当我们引用第三方库中的类需要装配到 Spring
容器时,则只能通过 @Bean
来实现。
@Bean
注解使用示例:
- @Configuration
- public class AppConfig {
- @Bean
- public TransferService transferService() {
- return new TransferServiceImpl();
- }
-
- }
上面的代码相当于下面的 xml 配置
- <beans>
- <bean id="transferService" class="com.acme.TransferServiceImpl"/>
- beans>
下面这个例子是通过 @Component
无法实现的。
- @Bean
- public OneService getService(status) {
- case (status) {
- when 1:
- return new serviceImpl1();
- when 2:
- return new serviceImpl2();
- when 3:
- return new serviceImpl3();
- }
- }
Spring 内置的 @Autowired
以及 JDK 内置的 @Resource
和 @Inject
都可以用于注入 Bean。
Annotaion | Package | Source |
---|---|---|
@Autowired | org.springframework.bean.factory | Spring 2.5+ |
@Resource | javax.annotation | Java JSR-250 |
@Inject | javax.inject | Java JSR-330 |
@Autowired
和@Resource
使用的比较多一些。
Autowired
属于 Spring 内置的注解,默认的注入方式为byType
(根据类型进行匹配),也就是说会优先根据接口类型去匹配并注入 Bean (接口的实现类)。
这会有什么问题呢? 当一个接口存在多个实现类的话,byType
这种方式就无法正确注入对象了,因为这个时候 Spring 会同时找到多个满足条件的选择,默认情况下它自己不知道选择哪一个。
这种情况下,注入方式会变为 byName
(根据名称进行匹配),这个名称通常就是类名(首字母小写)。就比如说下面代码中的 smsService
就是我这里所说的名称,这样应该比较好理解了吧。
- // smsService 就是我们上面所说的名称
- @Autowired
- private SmsService smsService;
举个例子,SmsService
接口有两个实现类: SmsServiceImpl1
和 SmsServiceImpl2
,且它们都已经被 Spring 容器所管理。
- // 报错,byName 和 byType 都无法匹配到 bean
- @Autowired
- private SmsService smsService;
- // 正确注入 SmsServiceImpl1 对象对应的 bean
- @Autowired
- private SmsService smsServiceImpl1;
- // 正确注入 SmsServiceImpl1 对象对应的 bean
- // smsServiceImpl1 就是我们上面所说的名称
- @Autowired
- @Qualifier(value = "smsServiceImpl1")
- private SmsService smsService;
我们还是建议通过 @Qualifier
注解来显示指定名称而不是依赖变量的名称。
@Resource
属于 JDK 提供的注解,默认注入方式为 byName
。如果无法通过名称匹配到对应的 Bean 的话,注入方式会变为byType
。
@Resource
有两个比较重要且日常开发常用的属性:name
(名称)、type
(类型)。
- public @interface Resource {
- String name() default "";
- Class> type() default Object.class;
- }
如果仅指定 name
属性则注入方式为byName
,如果仅指定type
属性则注入方式为byType
,如果同时指定name
和type
属性(不建议这么做)则注入方式为byType
+byName
。
- // 报错,byName 和 byType 都无法匹配到 bean
- @Resource
- private SmsService smsService;
- // 正确注入 SmsServiceImpl1 对象对应的 bean
- @Resource
- private SmsService smsServiceImpl1;
- // 正确注入 SmsServiceImpl1 对象对应的 bean(比较推荐这种方式)
- @Resource(name = "smsServiceImpl1")
- private SmsService smsService;
简单总结一下:
@Autowired
是 Spring 提供的注解,@Resource
是 JDK 提供的注解。
Autowired
默认的注入方式为byType
(根据类型进行匹配),@Resource
默认注入方式为 byName
(根据名称进行匹配)。
当一个接口存在多个实现类的情况下,@Autowired
和@Resource
都需要通过名称才能正确匹配到对应的 Bean。Autowired
可以通过 @Qualifier
注解来显示指定名称,@Resource
可以通过 name
属性来显示指定名称。