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

  首先,梳理Spring中的byName与byType的基本概念;然后,介绍@Resource和@Autowired的使用方法,同时介绍了@Primary注解;最后,介绍二者的相同点和区别。

byName与byType的概念

  首先,梳理一下基本概念,初步了解什么是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:按照名称(name)进行注入。该注解属于JDK中的注解,它在项目中使用频率比较高。

  @Autowired:按照类型自动装配。默认情况下要求依赖对象必须存在,如果要允许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提供,需要JDK6及以上。

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

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

  (4)构造器:@Autowired 可以写在构造器上,用于注入bean,@Resource不可以。

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

结束语

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

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

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

上一篇:12、手把手教你Extjs5(十二)执行菜单命令在tabPanel中显示模块


下一篇:impala 四舍五入后转换成string后又变成一个double的数值解决(除不尽的情况)