@Resource和@Autowired的区别
@Resource和@Autowired的不同点:
(1)基因不同: 首先其最大的不同在于她们的爸爸妈妈不同(@Autowired是由org.springframework.beans.factory.annotation.Autowired提供,换句话说就是由Spring提供;@Resource是由javax.annotation.Resource提供,即J2EE提供,需要JDK1.6及以上。)
(2)注入方式:@Autowired默认按照byType 注入,也提供byName;@Resource默认按byName自动注入,也提供按照byType 注入;
解释一波:顾名思义,byType是通过类型进行装配,byName是通过名称进行装配。
类型装配:根据要装配的变量类进行查找,如下
Spring会自动根据BWM类去IOC容器中寻找他的实现类或子类进行装配(如果出现两个实现类怎么办??别急,往下看,下面会讲)。
名称装配:根据之前在"Spring-context.xml"配置的id进行查找,如果用的"@Service(“id名”)"注解配置的,如果"name"没写,则默认name会定义成非全限定类名(首字母小写,如:类名为"BWM"则"name"会默认为"bWM"),此时"@Autowired"和"@Resource"的默认名称就是定义的属性名 。
那么你肯定想到了,如果我@Service或者xml的id中是我自己定义的id名,那我该如何配置呢?Spring已经想到了,看下面代码写法:
/* 比如你把BWM类的id定义成了bm,Benz类的id定义成了bc */ @AutoWired @Qualifier("bm") private BWM bwm; @Resource(name="bc") private Benz benz;
这就是按照名称装配了。
总结:
@Autowired按类型装配依赖对象,默认情况下它要求依赖对象必须存在,如果允许null值,可以设置它required属性为false。如果查询的结果不止一个,那么@Autowired会根据名称来查找。如果我们想使用按名称装配,也可以结合@Qualifier注解一起使用。
@Resource有两个中重要的属性:name和type。name属性指定byName,如果没有指定name属性,当注解标注在字段上,即默认取字段的名称作为bean名称寻找依赖对象,当注解标注在属性的setter方法上,即默认取属性名作为bean名称寻找依赖对象。需要注意的是,@Resource如果没有指定name属性,并且按照默认的名称仍然找不到依赖对象时, @Resource注解会回退到按类型装配。但一旦指定了name属性,就只能按名称装配了。
@Resource装配顺序
- 如果同时指定了name和type,则从Spring上下文中找到唯一匹配的bean进行装配,找不到则抛出异常
- 如果指定了name,则从上下文中查找名称(id)匹配的bean进行装配,找不到则抛出异常
- 如果指定了type,则从上下文中找到类型匹配的唯一bean进行装配,找不到或者找到多个,都会抛出异常
- 如果既没有指定name,又没有指定type,则自动按照byName方式进行装配;如果没有匹配,则回退为一个原始类型进行匹配,如果匹配则自动装配;
- 最后推荐使用@Resource注解在字段上,这样就不用写setter方法了.并且这个注解是属于J2EE的,减少了与Spring的耦合,这样代码看起就比较优雅 。