前言
开发的时候一直用@Autowired和@Resource注解实现自动装配,但是一直不明白byType这个装配方式是什么,研究才明白这两者之间的区别。
一、byName
@Resource注解一般是先根据名称去查找,如果没有再根据类型查找
一般Spring用@Resource注解实现byName,这个注解比较简单理解,就是用对象名注册到Spring容器中,如果出现相同名称项目启动就会报错。
二、byType
1.Class类型
首先咱们先理解什么是Class类型:
每个类的运行时的类型信息就是用Class对象表示的。它包含了与类有关的信息。其实我们的实例对象就通过Class对象来创建的。Java使用Class对象执行其RTTI(运行时类型识别:Run-Time
Type Identification),多态就是基于RTTI实现的。
每一个类都有一个Class对象,每当编译一个新类就产生一个Class对象,基本类型 (boolean,
int等)有Class对象,数组有Class对象,就连关键字void也有Class对象(void.class)。Class对象对应着java.lang.Class类,如果说类是对象抽象和集合的话,那么Class类就是对类的抽象和集合。
2.@Autowired
当你用一个接口,并且有多个实现类,同一个方法会有多种不同的实现逻辑。
接口Car
public interface Car {
int find();
int out();
}
实现类Benz
import org.springframework.stereotype.Service;
@Service
public class Benz implements Car{
@Override
public int find() {
System.out.println("奔驰");
return 0;
}
@Override
public int out() {
return 0;
}
}
实现类Honda
import org.springframework.stereotype.Service;
@Service
public class Honda implements Car{
@Override
public int find() {
System.out.println("本田");
return 0;
}
@Override
public int out() {
return 0;
}
}
控制层
import com.example.mqredises.entity.Car;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
@RequestMapping("/car")
public class CarController {
@Autowired
private Car car;
@RequestMapping("/test")
public int test(){
return car.find();
}
}
这个时候编译器会报错,显示
Could not autowire. There is more than one bean of ‘Car’ type.
Beans:
benz (Benz.java) honda (Honda.java)
会找到多个相同Car类型的Bean
这个时候咱们有两种解决方案:
- 使用@Qualifier,里面填具体的Bean名称,这样就也是转化成byName的方式去自动装配对象。
@Qualifier("benz")
@Autowired
private Car car;
- 改成用上文的@Resource注解,如果没指定名称,这个时候编译能通过但是运行会报错,所以也是需要指定对应的Bean名称
@Resource(name = "benz")
private Car car;
总结
实际上Spring的byName和byType基本上是不会出错的,都是可用的,只是有时候会出现不同包名底下有相同的类名,导致注册到Spring容器里报错了,也就是byName;而byType也是直接指定了接口的类型但底下却有多个实现类也会造成报错,实际开发过程中,一般都是直接指定实现类的类型,这样也不会出现报错;根据自己的项目架构选择合适的注解即可。