一、Autowired与Resource基本背景
1、两者都是用于bean的注入使用
2、@Resource
默认按照byName方式进行装配,属于J2EE自带注解,没有指定name时,name指的是变量名
1)如果同时指定了name和type,则从Spring上下文中找到唯一匹配的bean进行装配,找不到则抛出异常
2) 如果指定了name,则从上下文中查找名称(id)匹配的bean进行装配,找不到则抛出异常
3) 如果指定了type,则从上下文中找到类型匹配的唯一bean进行装配,找不到或者找到多个,都会抛出异常
4) 如果既没有指定name,又没有指定type,则自动按照byName方式进行装配;如果没有匹配,则回退为一个原始类型进行匹配,如果匹配则自动装配;
3、@Autowired
默认按byType自动注入,是Spring的注解
1)@Autowired默认按类型装配(属于spring的),默认情况下必须要求依赖对象必须存在,如果要允许null值,可以设置它的required属性为false;
2)@Autowired按类型装配的过程中,如果发现找到多个bean,则又按照byName方式进行比对,如果还有多个,则报出异常;
二、接口只有一个实现类
在开发过程中,会存在@Autowired跟@Resouce混着用的情况。
public interface Animal {
String talk();
}
//注意:@Service 默认value=dog,首字母小写
@Service
public class Dog implements Animal {
@Override
public String talk() {
return "Dog talk:";
}
}
@RestController
@Controller(value = "/zoo")
@RequestMapping("/zoo")
public class ZooController {
@Resource
Animal animal;
@GetMapping("/talk")
public String animalTalk() {
return animal.talk();
}
}
上面用简单的代码示例,演示接口只有一个实现类的情况下,使用Resouce注解。
结果:显示正确
改动,将Resouce改为Autowired,结果也正常显示。
结论:在接口只有一个实现类的时候,@Autowired跟@Resouce可以替换使用
三、接口有多个实现类
Animal接口,现在有Dog跟Cat两个实现类
@Service
public class Cat implements Animal {
@Override
public String talk() {
return "Cat talk";
}
}
1、@Resouce注解
运行时会报错,错误信息为:
No qualifying bean of type 'cn.woonton.water.demo.Animal' available: expected single matching bean but found 2: cat,dog
由于Resouce优先使用name去匹配,找到了2个实现类,所以报出了异常
解决办法:
1、修改注入的注解,增加name
@RestController
@Controller(value = "/zoo")
@RequestMapping("/zoo")
public class ZooController {
@Resource(name = "dog")
//@Qualifier("dog")
Animal animal;
......
}
再运行,正常。
2、修改实现类的service的value
//@Service 默认value=dog,首字母小写
@Service("animal")
public class Dog implements Animal {
......
}
再运行,正常。
2、@Autowired注解
IDEA直接提示报错
![image](https://upload-images.jianshu.io/upload_images/1944843-8d8e2f4fa7e62d78.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
解决办法:
1、使用@Primary修饰实现类,告诉Spring,如果有多个实现类时,优先注入@Primary注解修饰的那个。
@Primary
@Service
public class Dog implements Animal {
......
}
再运行,正常。
2、使用@Qualifier注解
@RestController
@Controller(value = "/zoo")
@RequestMapping("/zoo")
public class ZooController {
@Autowired
@Qualifier("dog")
Animal animal;
......
}
再运行,正常。
四、片尾
byName,这个name是指类名?还是变量名?
public interface Human {
String run();
}
@Service
public class Man implements Human {
@Override
public String run() {
return "Man run";
}
}
新增一个service,然后修改controller中的变量名
@RestController
@Controller(value = "/zoo")
@RequestMapping("/zoo")
public class ZooController {
@Resource
Animal man;
......
}
运行报错,错误信息为
Bean named 'man' is expected to be of type 'cn.woonton.water.demo.Animal' but was actually of type 'cn.woonton.water.demo.Man'
抱着尝试的态度,发现此时的name指的是变量名。所以根据变量名找到的类跟实际类不符合,所以报错了。