@Autowired 与@Resource的区别(详细)

#@Autowired 与@Resource的区别html

spring不但支持本身定义的@Autowired注解,还支持几个由JSR-250规范定义的注解,它们分别是@Resource、@PostConstruct以及@PreDestroy。java

@Resource的做用至关于@Autowired,只不过@Autowired按byType自动注入,而@Resource默认按 byName自动注入罢了。spring

@Resource有两个属性是比较重要的,分是name和type,Spring将@Resource注解的name属性解析为bean的名字,而type属性则解析为bean的类型。因此若是使用name属性,则使用byName的自动注入策略,而使用type属性时则使用byType自动注入策略。若是既不指定name也不指定type属性,这时将经过反射机制使用byName自动注入策略。数据库

  • @Resource装配顺序
  1. 若是同时指定了name和type,则从Spring上下文中找到惟一匹配的bean进行装配,找不到则抛出异常
  2. 若是指定了name,则从上下文中查找名称(id)匹配的bean进行装配,找不到则抛出异常
  3. 若是指定了type,则从上下文中找到类型匹配的惟一bean进行装配,找不到或者找到多个,都会抛出异常
  4. 若是既没有指定name,又没有指定type,则自动按照byName方式进行装配;若是没有匹配,则回退为一个原始类型进行匹配,若是匹配则自动装配;
  • @Autowired 与@Resource的区别:
  1. @Autowired与@Resource均可以用来装配bean. 均可以写在字段上,或写在setter方法上。session

  2. @Autowired默认按类型装配(这个注解是属业spring的),默认状况下必需要求依赖对象必须存在,若是要容许null值,能够设置它的required属性为false,如:@Autowired(required=false) ,若是咱们想使用名称装配能够结合@Qualifier注解进行使用,以下:架构

    @Autowired()@Qualifier("baseDao")
    	privateBaseDao baseDao;
  3. @Resource(这个注解属于J2EE的),默认按照名称进行装配,名称能够经过name属性进行指定,若是没有指定name属性,当注解写在字段上时,默认取字段名进行安装名称查找,若是注解写在setter方法上默认取属性名进行装配。当找不到与名称匹配的bean时才按照类型进行装配。可是须要注意的是,若是name属性一旦指定,就只会按照名称进行装配。app

    @Resource(name="baseDao")
    	privateBaseDao baseDao;

推荐使用:@Resource注解在字段上,这样就不用写setter方法了,而且这个注解是属于J2EE的,减小了与spring的耦合。这样代码看起就比较优雅。框架

  • spring @Qualifier注解
  1. @Autowired是根据类型进行自动装配的。若是当Spring上下文中存在不止一个UserDao类型的bean时,就会抛出BeanCreationException异常;若是Spring上下文中不存在UserDao类型的bean,也会抛出BeanCreationException异常。咱们可使用@Qualifier配合@Autowired来解决这些问题。以下:函数

    1. 可能存在多个UserDao实例

    [java] view plaincopy在CODE上查看代码片派生到个人代码片ui

    @Autowired
    	@Qualifier("userServiceImpl")
    	public IUserService userService;

或者

[java] view plaincopy在CODE上查看代码片派生到个人代码片
````java
@Autowired
public void setUserDao(@Qualifier("userDao") UserDao userDao) {
	this.userDao = userDao;
}
````

这样Spring会找到id为userServiceImpl和userDao的bean进行装配。

1. 可能不存在UserDao实例

[java] view plaincopy在CODE上查看代码片派生到个人代码片
```java
@Autowired(required = false)
public IUserService userService
```

我的总结:

@Autowired//默认按type注入

@Qualifier("cusInfoService")//通常做为@Autowired()的修饰用

@Resource(name="cusInfoService")//默认按name注入,能够经过name和type属性进行选择性注入

通常@Autowired和@Qualifier一块儿用,@Resource单独用。固然没有冲突的话@Autowired也能够单独用

-----------经常使用注解--------

--定义Bean的注解

@Controller

@Controller("Bean的名称")

定义控制层Bean,如Action

@Service

@Service("Bean的名称")

定义业务层Bean

@Repository

@Repository("Bean的名称")

定义DAO层Bean

@Component

定义Bean, 很差归类时使用.

自动装配Bean (选用一种注解就能够)

@Autowired (Srping提供的)

默认按类型匹配,自动装配(Srping提供的),能够写在成员属性上,或写在setter方法上

@Autowired(required=true)

必定要找到匹配的Bean,不然抛异常。 默认值就是true

@Autowired

@Qualifier("bean的名字")

按名称装配Bean,与@Autowired组合使用,解决按类型匹配找到多个Bean问题。

@Resource JSR-250提供的

默认按名称装配,当找不到名称匹配的bean再按类型装配.

能够写在成员属性上,或写在setter方法上

能够经过@Resource(name="beanName") 指定被注入的bean的名称, 要是未指定name属性, 默认使用成员属性的变量名,通常不用写name属性.

@Resource(name="beanName")指定了name属性,按名称注入但没找到bean, 就不会再按类型装配了.

@Inject 是JSR-330提供的

按类型装配,功能比@Autowired少,没有使用的必要。

--定义Bean的做用域和生命过程

@Scope("prototype")

值有:singleton,prototype,session,request,session,globalSession

@PostConstruct

至关于init-method,使用在方法上,当Bean初始化时执行。

@PreDestroy

至关于destory-method,使用在方法上,当Bean销毁时执行。

--声明式事务

@Transactional

@Autowired @Resource @Qualifier的区别

实用理解:@Autowired @Resource 二选其一,看中哪一个就用哪一个。

简单理解:

@Autowired 根据类型注入,

@Resource 默认根据名字注入,其次按照类型搜索

@Autowired @Qualifie("userService") 两个结合起来能够根据名字和类型注入

复杂理解:

好比你有这么一个Bean

@Service(“UserService”)

public Class UserServiceImpl implements UserService{};

如今你想在UserController 里面使用这个UserServiceImpl

public Class UserController {

@AutoWire //当使用这个注入的时候上面的 UserServiceImpl 只须要这样写 @Service,这样就会自动找到UserService这个类型以及他的子类型。UserServiceImpl 实现了UserService,因此可以找到它。不过这样有一个缺点,就是当UserService实现类有两个以上的时候,这个时候会找哪个呢,这就形成了冲突,因此要用@AutoWire注入的时候要确保UserService只有一个实现类。

@Resource 默认状况下是按照名称进行匹配,若是没有找到相同名称的Bean,则会按照类型进行匹配,有人可能会想了,这下好了,用这个是万能的了,不用管名字了,也不用管类型了,但这里仍是有缺点。首先,根据这个注解的匹配效果能够看出,它进行了两次匹配,也就是说,若是你在UserService这个类上面这样写注解,@Service,它会怎么找呢,首先是找相同名字的,若是没有找到,再找相同类型的,而这里的@Service没有写名字,这个时候就进行了两次搜索,显然,速度就降低了许多。也许你还会问,这里的@Service原本就没有名字,确定是直接进行类型搜索啊。其实不是这样的,UserServiceImpl 上面若是有@Service默认的名字 是这个userServiceImpl,注意看,就是把类名前面的大写变成小写,就是默认的Bean的名字了。 @Resource根据名字搜索是这样写@Resource("userService"),若是你写了这个名字叫userService,那么UserServiceImpl上面必须也是这个名字,否则仍是会报错。

@Autowired @Qualifie("userService") 是直接按照名字进行搜索,也就是说,对于UserServiceImpl 上面@Service注解必须写名字,不写就会报错,并且名字必须是@Autowired @Qualifie("userService") 保持一致。若是@Service上面写了名字,而@Autowired @Qualifie() ,同样会报错。

private UserService userService;

说了这么多,可能你有些说晕了,那么怎么用这三个呢,要实际的工做是根据实际状况来使用的,一般使用AutoWire和@Resource多一些,bean的名字不用写,而UserServiceImpl上面能会这样写 @Service("userService")。这里的实际工做状况,究竟是什么状况呢?说白了就是整个项目设计时候考虑的状况,若是你的架构设计师考虑的比较精细,要求比较严格,要求项目上线后的访问速度比较好,一般是考虑速度了。这个时候@AutoWire没有@Resource好用,由于@Resource能够根据名字来搜索,是这样写的@Resource("userService")。这个@Autowired @Qualifie("userService") 也能够用名字啊,为何不用呢,缘由很简单,这个有点长,不喜欢,增长工做量。由于根据名字搜索是最快的,就好像查数据库同样,根据Id查找最快。由于这里的名字与数据库里面的ID是同样的做用。这个时候,就要求你多写几个名字,工做量天然就增长了。而若是你不用注解,用xml文件的时候,对于注入Bean的时候要求写一个Id,xml文件时候的id就至关于这里的名字。

说了那么多没用,你能作的就是简单直接,什么最方便就用什么,

你就直接用@Resource得了,若是你喜欢用@AutoWire也行,不用写名字。

一般状况一个Bean的注解写错了,会报下面这些错误,最为常见,

No bean named 'user' is defined,这个表示没有找到被命名为user的Bean,通俗的说,就是名字为user的类型,以及它的子类型,出现这个错误的缘由就是注入时候的类型名字为user,而搜索的时候找不到,也就是说可能那个搜索的类型,并无命令为user,解决办法就是找到这个类型,去命令为user,

下面这个错误也常见,

No qualifying bean of type [com.service.UserService] found for dependency:

这个错误的缘由就是类型上面没有加@Service这个注入,不只仅是@Service,若是是其余层也会出现这个错误,这里我是以Service为例子说明,若是是DAO层就是没有加@Repository,Controller层,则是没有加@Controller。

还有,若是你仍是想再简单点,不管是DAO,Controller,Service三个层,均可以用这个注解,@Component,这个注解通用全部的Bean,这个时候你可能会说了,有一般的为何用的人少呢,那是由于MVC这个分层的设计原则,用@Repository,@Service,@Controller,这个能够区别MVC原则中的DAO,Service,Controller。便于识别。

博客2:

springautowiredqualifierbytypebyname  在使用Spring框架中@Autowired标签时默认状况下使用 Java代码 @Autowired
@Autowired 注释进行自动注入时,Spring 容器中匹配的候选 Bean 数目必须有且仅有一个。当找不到一个匹配的 Bean 时,Spring 容器将抛出 BeanCreationException 异常,并指出必须至少拥有一个匹配的 Bean。 @Autowired 默认是按照byType进行注入的,若是发现找到多个bean,则,又按照byName方式比对,若是还有多个,则报出异常。

例子:

@Autowired private ExamUserMapper examUserMapper; - ExamUserMapper是一个接口

  1. spring先找类型为ExamUserMapper的bean

  2. 若是存在且惟一,则OK;

  3. 若是不惟一,在结果集里,寻找name为examUserMapper的bean。由于bean的name有惟一性,因此,到这里应该能肯定是否存在知足要求的bean了

@Autowired也能够手动指定按照byName方式注入,使用@Qualifier标签,例如: @Autowired () @Qualifier ( "baseDao" )

  Spring 容许咱们经过 Java代码 @Qualifier
@Qualifier 注释指定注入 Bean 的名称,这样歧义就消除了,能够经过下面的方法解决异常。

   Java代码 @Qualifier("XXX")
@Qualifier("XXX") 中的 XX是 Bean 的名称,因此 @Autowired 和 @Qualifier 结合使用时,自动注入的策略就从 byType 转变成 byName 了。

  @Autowired 能够对成员变量、方法以及构造函数进行注释,而 @Qualifier 的标注对象是成员变量、方法入参、构造函数入参。

Spring不但支持本身定义的@Autowired注解,还支持几个由JSR-250规范定义的注解,它们分别是@Resource、@PostConstruct以及@PreDestroy。

   Java代码 @Resource
@Resource 的做用至关于@Autowired,只不过@Autowired按byType自动注入,而@Resource默认按 byName自动注入罢了。@Resource有两个属性是比较重要的,分是name和type,Spring将@Resource注解的name属性解析为bean的名字,而type属性则解析为bean的类型。因此若是使用name属性,则使用byName的自动注入策略,而使用type属性时则使用byType自动注入策略。若是既不指定name也不指定type属性,这时将经过反射机制使用byName自动注入策略。

  @Resource装配顺序   1. 若是同时指定了name和type,则从Spring上下文中找到惟一匹配的bean进行装配,找不到则抛出异常   2. 若是指定了name,则从上下文中查找名称(id)匹配的bean进行装配,找不到则抛出异常   3. 若是指定了type,则从上下文中找到类型匹配的惟一bean进行装配,找不到或者找到多个,都会抛出异常   4. 若是既没有指定name,又没有指定type,则自动按照byName方式进行装配;若是没有匹配,则回退为一个原始类型进行匹配,若是匹配则自动装配

参考博文:

http://www.javashuo.com/article/p-cfzanhru-g.html

http://www.cnblogs.com/happyyang/articles/3553687.html

http://blog.csdn.net/revent/article/details/49203619

http://blog.csdn.net/ad921012/article/details/49679745

相关文章
相关标签/搜索