自动装配组件@Resource和@Autowired的区别

  @Resource和@Autowired这两个注解都提供了将依赖对象注入到当前对象的功能,但二者却有众多区别,并且它们是常见的面试题之一,所以楼兰胡杨今天就来梳理它,希望对各位有所帮助。首先,梳理Spring中的byName与byType的基本概念;然后,介绍@Resource和@Autowired的使用方法,同时介绍了@Primary注解;最后,介绍二者的相同点和区别。

byName与byType的概念

  依赖注入是通过先在 Spring IoC 容器中查找对象,再将对象作为属性注入到当前类中。而查找的方式有两种:按名称(byName)查找或按类型(byType)查找,其中 @Autowired 和 @Resource 都是既使用了名称查找又使用了类型查找,但二者进行查找的顺序却截然相反。

  首先,梳理一下基本概念,初步了解什么是byType,什么是byName。

<bean id="userServiceImpl" class="cn.com.service.impl.UserServiceImpl" Autowire="byName">
</bean>
<bean id="userDao" class="cn.com.dao.impl.UserDaoImpl">
</bean>

  比如说上面这段XML代码,autowire="byType"的意思是通过class="cn.com.dao.impl.UserDaoImpl"来查找UserDaoImpl下所有的对象,代码autowire="byName"意思是通过id="userDao"来查找Bean中的userDao对象。

  byName就是通过Bean的id或者name去自动装配,如果一个bean的name 和另外一个bean的 property 相同,就自动装配。byType就是按Bean的Class的类型去自动装配,如果一个bean的数据类型和另外一个bean的property属性的数据类型兼容,就自动装配。

使用方法

   @Resource:先按照名称进行注入,如果按照名称查找不到,再根据类型进行查找。该注解属于JDK中的注解,它在项目中使用频率比较高。

  @Autowired:先按照类型(byType)自动装配,如果存在多个相同类型的 Bean 再按照名称(byName)自动装配。默认情况下要求依赖对象必须存在,如果要允许null值,可以设置它的required属性为false,required属性的默认值为true。例如:

@Autowired(required=false) 
@Qualifier("userService")
private UserService userService;

   如果当Spring上下文中存在不止一个UserDao类型的组件时,就会抛出BeanCreationException异常,为了解决这个问题,@Autowired注解可以借助
@Qualifier("组件名")注解来指明使用哪一个组件(实现类),实际上这是通过byName的方式实现。@Qualifier注解是java的规范,JRE标准

  @Primary:和@Qualifier 一样,可以使用@Primary注解让Spring进行自动装配的时候,默认使用哪个组件。使用场景经常是:对同一个接口,可能会有几种不同的实现类,而默认只会采取其中一种的情况下, @Primary 就会大显身手。

  例如,UserService有两个实现类UserServiceImpl和UserServiceImplDefault,而我们想默认使用后者,可以使用@Primary修饰UserServiceImplDefault类:


@Service
public class UserServiceImpl implements UserService {
    private static Logger logger = LoggerFactory.getLogger(UserServiceImpl.class);

    @Autowired
    private UserDao userDao;
    @Override
    public User getUserById(Long userId) {
        logger.info("--------*****-----------");
        return userDao.getUserById(userId);
    }
}

// ----------------- 我是分割线 -----------------

@Primary
@Component("defaultUserBean")
public class UserServiceImplDefault implements UserService {
    private static Logger logger = LoggerFactory.getLogger(UserServiceImplDefault.class);

    @Autowired
    private UserDao userDao;
    @Override
    public User getUserById(Long userId) {
        logger.info("--------优先加载的组件-----------");
        return userDao.getUserById(userId);
    }
}

   @Resource,默认按照名称进行装配,名称可以通过name属性进行指定,如果没有指定name属性,当注解写在字段上时,默认取字段的名称作为bean名称寻找依赖对象,当注解标注在属性的setter方法上时,默认取属性名作为bean名称寻找依赖对象。需要注意的是,@Resource如果没有指定name属性,并且按照默认的名称仍然找不到依赖对象时, 它会回退到按类型装配。但是需要格外注意的是,一旦用name指定加载的组件名称,就只能按名称自动装配。

@Resource (name= "userService" )
private UserService userService;

  @Inject:和@Autowired注解一样也是按类型注入bean,但是没有required属性。使用此注解时需要导入javax.inject包。

  以上三个注解主要用于为类中属性自动注入bean,它们可以将(Spring IOC容器中的对象)装配到某些类的属性中。

Spring注解实战

  例如,使用@Repository注册UserDao的组件到Spring容器:

@Repository
public interface UserDao {
    User getUserById(Long userId);
}

  使用@Autowired把组件UserDao装配到UserServiceImpl实现类中:

@Service
public class UserServiceImpl implements UserService {
    private static Logger logger = LoggerFactory.getLogger(UserServiceImpl.class);

    @Autowired
    private UserDao userDao;
    @Override
    public User getUserById(Long userId) {
        logger.info("--------*****-----------");
        return userDao.getUserById(userId);
    }
}

  这里,@Service将UserServiceImpl注册成组件托管给Spring容器。

相同点与区别

  相同点:@Resource的作用相当于@Autowired,均可标注在字段或属性的setter方法上。

  不同点

  (1)提供方:@Autowired是由Spring提供,即由org.springframework.beans.factory.annotation.Autowired提供;@Resource是由JDK提供,即由javax.annotation.Resource提供,来自于 JSR-250,需要JDK 6及以上版本。

  (2)注入方式:@Autowired只按照byType 注入;可以借助@Qualifier注解来指明使用哪一个实现类,@Resource默认按byName自动注入,也提供按照byType 注入。

  (3)注解属性:@Autowired 注解只支持设置一个 required 属性,而 @Resource 注解支持七种属性。@Autowired 按类型装配依赖对象,默认情况下它要求依赖对象必须存在,如果允许null值,可以设置它required属性为false。如果我们想使用按名称装配,可以结合@Qualifier注解一起使用。

  (4)依赖注入的支持不同:@Autowired 可以写在构造器上,用于注入bean,@Resource不可以。@Autowired 支持三种常见依赖注入方式——属性注入、构造方法注入和 Setter 注入,而 @Resource 只支持属性注入和 Setter 注入。

  (5)@Resource注解不支持spring的@Primary注解优先注入,但是@Autowired支持。

结束语

  @Autowied、@Resource或@Inject什么时候实现自动装配?启动spring IoC时,容器自动装载了一个AutowiredAnnotationBeanPostProcessor后置处理器,当容器扫描到@Autowied、@Resource或@Inject时,就会在IoC容器自动查找需要的组件,并装配给该对象的属性。

  推荐使用@Resource注解,使代码更加优雅,用它修饰字段后,就不用写setter方法了,并且这个注解是属于J2EE的,减少了与Spring的耦合。

  作为程序员,要有“刨根问底”的精神。知其然,更要知其所以然。这篇文章希望能抽丝剥茧,还原背后的原理。

posted @ 2021-09-21 15:18  楼兰胡杨  阅读(396)  评论(0编辑  收藏  举报