相同点:
@Resource和@Autowired,都用来完成对bean的注入;都可以写在字段上,或者是写在setter方法上
//两者如果都写在字段上,那么就不需要再写setter方法。写在setter方法上的也不多见
@Resource(name="userDao")
private UserDao userDao;
@Autowired
private ProductDao productDao;
不同点:
①@Autowired,属于Spring的注解,来自org.springframework.beans.factory.annotation.Autowired包
@Resource,不属于Spring的注解,是JDK1.6支持的注解,来自javax.annotation.Resource包
②@Autowired,默认按byType进行匹配,如果发现有多个bean,则(使用 @Qualifier 注解解决)按照 byName 进行匹配,如果还是有多个,则会报异常
@Resource,默认按byName自动注入
@Resource用法
1.首先,启动Spring容器
2.启动Spring容器后,Spring会默认去扫描容器中有哪些bean可以用来加载(xml和注解都会被扫描到),会扫描到bean实体类的属性和set方法上是否有@Resource的注解
3.如果在属性中找到@Resource注解后
①@Resource(name="xxx")。
如果@Resource注解括号中有name属性,属性值不为空,则会在Spring容器中扫描是否存在于该name属性值相同的bean,如果存在,则注入成功;如若找不到,则注入失败,抛出异常。
②@Resource 或者 @Resource(name="")。
如果@Resouce注解括号中没有name属性,或者name属性为空。则Spring会扫描容器中是否存在与该属性名称相同的bean存在,如若存在,则注入成功。如不存在,则会继续去扫描,看Spring容器中bean的id所对应的类型是否与@Resource注解的该变量所对应的类型是否相等,如果相等,则注入成功,否则注入失败,抛出异常
一、@Resource注解的name属性不为空
首先创建Person类,并纳入容器中管理:
@Autowired注解是按照类型(byType)装配依赖对象
package com.lzj.springboot.resource;
import org.springframework.stereotype.Component;
/*纳入容器中后,bean的id名字为ps*/
@Component(value="ps")
public class Person {
public void say(){
System.out.println("------say()------");
}
}
然后创建Man类,类中的属性依赖Person类型的bean
package com.lzj.springboot.resource;
import javax.annotation.Resource;
import org.springframework.stereotype.Component;
@Component
public class Man {
/*从容器中取id名字为ps的bean,如果找不到该bean,spring启动过程中就会报错,表示把Man类型的bean注入到容器中不成功,因为person的属性依赖注入的时候就出错了,所以创建Man的bean的时候肯定不成功。*/
@Resource(name="ps")
private Person person;
/*依赖注入失败,因为Person类型注入到容器中的bean的id指定为ps,所以从容器中获取id为person的bean就会失败*/
// @Resource(name="person")
// private Person ps;
public void work(){
person.say();
System.out.println("------work()------");
}
}
启动类为:
@SpringBootApplication(scanBasePackages="com.lzj.springboot")
public class App {
public static void main(String[] args) {
SpringApplication app = new SpringApplication(App.class);
ConfigurableApplicationContext context = app.run(args);
/*从容器中获取Man类型的bean,如果Man类型的bean注入到容器成功,此时就能获取到;如果注入不成功,则获取不到。注入不成功,就会有可能是Man中的@Resource注解的依赖注入没有成功*/
context.getBean(Man.class).work();
context.close();
}
}
启动工程,输出如下:
------say()------
------work()------
二、@Resource注解的name属性为空
1、@Resource要注解的那个变量属性与容器中的bean的id的名字相等
启动类和Person的类与相面一样,下面直接修改Man类如下:
@Component
public class Man {
/*@Resource注解的属性变量ps与容器中的bean的id名字ps相等,可以匹配*/
@Resource
private Person ps;
public void work(){
ps.say();
System.out.println("------work()------");
}
}
2、@Resource要注解的那个变量属性与容器中的bean的id的名字不相等
启动类和Person的类与相面一样,下面直接修改Man类如下:
@Component
public class Man {
/*@Resource注解的属性变量person与容器中的bean的id名字ps不相等,然后通过bean的类型判断:person变量属性的类型为Person类,容器中的id为ps的bean的类型也为Person类型,因此此种情况下也可以匹配*/
@Resource
private Person person;
public void work(){
person.say();
System.out.println("------work()------");
}
}
@Autowired用法
①默认按类型装配
②依赖对象必须存在,如果要允许null值,可以设置它的required属性为false @Autowired(required=false)
③也可以使用名称装配,配合@Qualifier注解
public class TestServiceImpl {
// 下面两种@Autowired只要使用一种即可
@Autowired
private UserDao userDao; // 用于字段上
@Autowired
public void setUserDao(UserDao userDao) { // 用于属性的方法上
this.userDao = userDao;
}
}
一、下面再介绍@Autowired 注解的其它几种使用情况
1、@Autowired 注解的 required 属性
默认情况下,所有使用@Autowired 注解的属性,该属性对应的实例一定能在IOC容器中能找到才可以。如果IOC容器中不存在,则会抛出找不到 bean 的实例错误。因为@Autowired 的required 属性默认为true,表示属性对应的bean的实例一定要存在才可以。如果要IOC容器中不存在属性对应的bean的实例,在程序运行的时候也不报错,可以把required 置为false。
2、@Autowired 注解配合@Qualifier使用
当用@Autowired为属性自动注入bean的实例时,如果IOC容器中有多个bean的实例,程序就会抛一个IOC中没有一个唯一bean实例的错误。例如:
package com.lzj.spring;
public interface Animal
public void eat();
}
package com.lzj.spring;
import org.springframework.stereotype.Component;
@Component(value="dog")
public class Dog implements Animal {
@Override
public void eat() {
System.out.println("狗吃狗食。。。");
}
}
package com.lzj.spring;
import org.springframework.stereotype.Component;
@Component(value="cat")
public class Cat implements Animal {
@Override
public void eat() {
System.out.println("猫吃猫食。。。");
}
}
package com.lzj.spring.component;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.stereotype.Component;
import com.lzj.spring.Animal;
import com.lzj.spring.controller.ControllerBean;
@Component("beanComponent")
public class ComponentBean {
/*把IOC容器中Animal 类型的实例注入到animal属性中。注意:Animal是一个接口,所有实现该接口的类的实例都可以注入到animal属性中。*/
@Autowired
private Animal animal;
public void show() {
System.out.println("ComponentBean........");
animal.eat();
}
}
当运行测试类时,会抛出一个没有唯一bean实例的错误,因为此时IOC容器中有dog和cat的实例,两者都是Animal的实例。此时就可以用@Qualifier注解指定具体使用哪个IOC容器中哪个实例了。例如:
@Component("beanComponent")
public class ComponentBean {
/*Qualifier注解指定把IOC容器中dog实例注入到animal属性中*/
@Autowired
@Qualifier("dog")
private Animal animal;
public void show() {
System.out.println("ComponentBean........");
animal.eat();
}
}
运行测试类,输出:
ComponentBean........
狗吃狗食。。。
总结:@Autowired自动注解,举个例子吧,一个类,俩个实现类,Autowired就不知道注入哪一个实现类。
@Resource有name属性,可以区分。