详解 Spring 中 Bean 的自动装配
自动装配是Spring满足bean依赖的一种方式
Spring会在上下文中自动寻找,并且自动给bean装配属性
1. 环境搭建
一个人有猫和狗两个宠物
结构图:
Cat.java
package pojo;
public class Cat {
public void shout() {
System.out.println("喵~");
}
}
Dog.java
package pojo;
public class Dog {
public void shout() {
System.out.println("汪~");
}
}
People.java
package pojo;
public class People {
private Cat cat;
private Dog dog;
private String name;
public void setCat(Cat cat) {
this.cat = cat;
}
public void setDog(Dog dog) {
this.dog = dog;
}
public void setName(String name) {
this.name = name;
}
public Cat getCat() {
return cat;
}
public Dog getDog() {
return dog;
}
public String getName() {
return name;
}
}
beans.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
https://www.springframework.org/schema/beans/spring-beans.xsd">
<bean id="cat" class="pojo.Cat"/>
<bean id="dog" class="pojo.Dog"/>
<bean id="people" class="pojo.People">
<property name="name" value="zsr"/>
<property name="dog" ref="dog"/>
<property name="cat" ref="cat"/>
</bean>
</beans>
MyTest.java
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import pojo.People;
public class MyTest {
public static void main(String[] args) {
ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
People people = context.getBean("people", People.class);
people.getDog().shout();
people.getCat().shout();
System.out.println(people.getName());
}
}
测试:
到此环境搭建成功!
2. byName、byType
上述代码中,ref就是寻找对应的cat和dog对象;如果有一种机制,能够自动在bean里面寻找我们要设置的cat或者dog,自动装配,就不用写下面的那两行代码了!这就引入了自动装配
我们更改一下.xml文件,删除那两行代码,加入autowire属性,发现有几种可以选择,这里先选择byName
<bean id="cat" class="pojo.Cat"/>
<bean id="dog1" class="pojo.Dog"/>
<bean id="people" class="pojo.People" autowire="byName">
<property name="name" value="zsr"/>
</bean>
再次运行测试类,依旧成功,说明我们的cat和dog被自动设置到了people中
那我们再更改一下,将dog改为dog1
<bean id="cat" class="pojo.Cat"/>
<bean id="dog1" class="pojo.Dog"/>
<bean id="people" class="pojo.People" autowire="byName">
<property name="name" value="zsr"/>
</bean>
再进行测试,发现空指针异常
为什么??
byName:根据属性名自动装配,检查IoC容器并根据名字查找与属性完全一致的bean,并将其与属性自动装配。
因此bean的id只有为dog或者cat才能够被找到
我们再改一下,将byName改为byType
<bean id="cat" class="pojo.Cat"/>
<bean id="dog" class="pojo.Dog"/>
<bean id="people" class="pojo.People" autowire="byType">
<property name="name" value="zsr"/>
</bean>
再次运行,又成功了
byType:根据属性类型自动装配
如果容器中存在一个与指定属性类型相同的bean,那么将与该属性自动装配
如果存在多个该类型bean,那么抛出异常,并指出不能使用byType方式进行自动装配;
如果没有找到相匹配的bean,则什么事都不发生
必须保证类型全局唯一,如果多加一个dog,则报错:
可以省略id
<bean class="pojo.Cat"/>
<bean class="pojo.Dog"/>
<bean id="people" class="pojo.People" autowire="byType">
<property name="name" value="zsr"/>
</bean>
小结
byName的时候,需要保证所有bean的id唯一,并且这个bean id需要和实体类中对应的属性名相同
byName的时候,需要保证所有bean的class唯一,并且这个bean id需要和自动注入属性的类型一致;可以省略id
3. 使用注解实现自动装配
jdk1.5支持注解,Spring2.5支持注解
要使用注解:
xml中导入约束
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans
https://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
https://www.springframework.org/schema/context/spring-context.xsd">
xml中配置注解支持
<context:annotation-config/>
@Autowired
直接在属性上使用即可,此时就可以省略set方法
也可以在set方法上使用(适用于在set方法里面写一些处理逻辑的情况)
beans.xml
<context:annotation-config/>
<bean id="cat1" class="pojo.Cat"/>
<bean id="dog1" class="pojo.Dog"/>
<bean id="people" class="pojo.People">
<property name="name" value="zsr"/>
</bean>
运行测试类,成功
自动装配环境很复杂的情况下:假设此时有两个cat和dog对象
<context:annotation-config/>
<bean id="cat1" class="pojo.Cat"/>
<bean id="cat2" class="pojo.Cat"/>
<bean id="dog1" class="pojo.Dog"/>
<bean id="dog2" class="pojo.Dog"/>
<bean id="people" class="pojo.People">
<property name="name" value="zsr"/>
</bean>
发现直接爆红了
这是因为无法匹配到对应的属性,@Autowired 默认通过 byType 的方式实现,如果有多个对象,则通过 byName 查找,如果都找不到,则报错;
此时通过有两个相同类型的对象,通过类型无法找到,bean id也不与属性名相同,通过名字也找不到,所以报错;
这时候就需要配合@Qualifier来使用,指定唯一的bean对象来注入
再次运行,即可成功
@Qualifier还可以设置required属性值为false 允许属性值为null
@Resource
我们将上述的Autowired换成Resource
xml文件为:
<context:annotation-config/>
<bean id="cat1" class="pojo.Cat"/>
<bean id="cat2" class="pojo.Cat"/>
<bean id="dog1" class="pojo.Dog"/>
<bean id="dog2" class="pojo.Dog"/>
<bean id="people" class="pojo.People">
<property name="name" value="zsr"/>
</bean>
直接运行
发现报错了,这是因为我们无法匹配到对应的属性,@Resource 默认通过 byName 的方式实现,如果找不到名字, 则通过 byType 实现!如果两个都找不到,就会报错
此时,名字cat1/cat2/dog1/dog2不匹配,且每个类型的bean不唯一,所以通过名字和类型都找不到,因此报错
如果我们将其中一个bean的id改为对应的属性名cat和dog
<bean id="cat" class="pojo.Cat"/>
<bean id="cat2" class="pojo.Cat"/>
<bean id="dog" class="pojo.Dog"/>
<bean id="dog2" class="pojo.Dog"/>
<bean id="people" class="pojo.People">
<property name="name" value="zsr"/>
</bean>
再次运行,成功了
这是因为我们通过名字找到了
我们还有另一种解决方式,不用更改bean的id,我们在@Resource里面添加name属性
通过设置name属性指定装配的对象
此时再运行,成功注入
小结
@Autowired和@Resource的相同点:
都是用于自动装配的,都可以放在属性字段和set方法上
@Autowired和@Resource的区别:
@Autowired为Spring提供的注解,@Resource为java提供的注解
@Autowired 默认通过 byType 的方式实现,如果有多个对象,则通过 byName 查找,如果都找不到,则报错;此时可以用@Qualifier指定唯一的bean对象
@Resource 默认通过 byName 的方式实现,如果找不到名字, 则通过 byType 实现!如果两个都找不到,就会报错
作者:Baret-H
出处:blog.csdn.net/qq_45173404/article/details/107826456
关注GitHub今日热榜,专注挖掘好用的开发工具,致力于分享优质高效的工具、资源、插件等,助力开发者成长!
点个在看,你最好看