自动装配:是Spring满足bean依赖的一种方式!
Spring会在上下文中自动寻找,并自动被bean装配属性!
在Spring中有三种装配的方式:
- 在xml中显示的配置
- 在java中显示配置
- 隐式的自动转配bean【重要】
1、byName自动装配
<bean id="cat" class="lin.pojo.Cat"/>
<bean id="dog" class="lin.pojo.Dog"/>
<bean id="people" class="lin.pojo.People" autowire="byName">
<property name="name" value="林XX"/>
</bean>
@Test
public void test1(){
ApplicationContext context =
new ClassPathXmlApplicationContext("applicationContext.xml");
People people = context.getBean("people", People.class);
System.out.println(people.getName());
people.getCat().shout();
people.getDog().shout();
}
此时 byNme 方法会自动在容器上下文中查找,和自己对象set方法后面值对应的beanid!
@Data
@NoArgsConstructor
@AllArgsConstructor
public class People {
private Cat cat;
private Dog dog;
private String name;
}
这时候在People这个实体类中,有dog与cat的引用类,并且xml配置文件中也有对应的beanid为cat,dog。然后autowire="byName"就会去自动装配!
2、byType自动装配
<bean id="cat" class="lin.pojo.Cat"/>
<bean id="dog1111" class="lin.pojo.Dog"/>
<bean id="people" class="lin.pojo.People" autowire="byType">
<property name="name" value="林XX"/>
</bean>
在相同的测试类的情况下,程序能够正常运行!所以可以得出结论:byType 也是会自动在容器上下文中查找,和自己对象属性类型相同的bean!
byType的弊端:
byType的弊端就是需要保证这个类型全局唯一,才可以自动装配!
byType的好处:
此时没有这两个需要自动装配的bean没有id,程序依然可以正常运行。因为byType主要是根据类型来自动装配的!
以上小结:
byName:需要保证bean的id唯一,并且这个bean的id需要和自动注入属性的set方法一致!
byTyep :需要保证bean的class唯一,并且这个bean需要和自动注入属性的类型一致!
3、使用注解实现自动装配【*】
大多数情况我们会使用注解的形式进行开发而不是xml,因为xml比较的繁琐!
要使用注解须知:
1)导入约束: context约束(context的上下文支持)
2)配置注解的支持: <context:annotation-config/>开启注解支持
<?xml version="1.0" encoding="UTF-8"?>
<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
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd">
<context:annotation-config/>
</beans>
@Autowired(<context:annotation-config/>必须要开启注解支持)
前提是名字要一致:例如People中cat与bean中的id为cat的对应----注解是使用反射来实现的
@Data
@NoArgsConstructor
@AllArgsConstructor
public class People {
@Autowired
private Cat cat;
@Autowired
private Dog dog;
private String name;
}
<bean id="cat" class="lin.pojo.Cat"/>
<bean id="dog" class="lin.pojo.Dog"/>
<bean id="people" class="lin.pojo.People"/>
<context:annotation-config/>
@Test
public void test1(){
ApplicationContext context =
new ClassPathXmlApplicationContext("applicationContext.xml");
People people = context.getBean("people", People.class);
System.out.println(people.getName());
people.getCat().shout();
people.getDog().shout();
}
这里有一个很常见的问题:java.lang.NullPointerException ----当 <context:annotation-config/>没加时,不支持注解支持
测试时就会报这个异常!
还有一个问题就是:若名字不一致时, @Autowired自动注入时,也会用类型去匹配,当名字与类型都匹配不到时,会报错!
【1】当名字不匹配时,去匹配类型的情况!这时候bean里面的id与实体类中的名字不一致,但是程序正常运行!
<bean id="cat1" class="lin.pojo.Cat"/>
<bean id="dog1" class="lin.pojo.Dog"/>
<bean id="people" class="lin.pojo.People"/>
public class People {
@Autowired
private Cat cat;
@Autowired
private Dog dog;
private String name;
}
【2】当名字与类型都找不到时: 这时实体类中会报错,程序不能正常运行!
<bean id="cat1" class="lin.pojo.Cat"/>
<bean id="cat11111" class="lin.pojo.Cat"/>
<bean id="dog1" class="lin.pojo.Dog"/>
<bean id="dog11111" class="lin.pojo.Dog"/>
<bean id="people" class="lin.pojo.People"/>
这时候自动装配无法实现了!
但是加了@Qualifier限定词注解之后,又能在bean中找到cat1与dog1,所以程序现在又能成功执行!
@Autowired 的使用注意点:
【1】直接在属性上使用即可,也可以在set方式上使用!
【2】使用@Autowired时,我们可以不用写set方法,前提是你这个自动装配在IOC容器中存在,且符合名字!
@Autowired(required = false) //说明这个对象可以为null,否则不允许为空@Nullable //字段标记了这个注解,说明这个字段可以为n
如果 @Autowired 自动装配的环境比较复杂,自动装配无法通过一个注解【 @Autowired 】完成的时候,我们可以用【@Qualifier(value = "XXX")】去配合@Autowired 使用,指定一个唯一的bean对象注入!
@Resources注解:
java也有一个注解可以实现自动装配功能:【@Resources】这个会先通过名字去查找bean,然后在通过类型去查找。要是都找不到时,使用【@Resources(name = "XXX")】去匹配bean中id对应XXX的bean
@Autowired和@Resources的区别:
【1】都是用来自动装配的,都可以放在属性字段上
【2】 @Autowired通过byType的方式实现,@Resources默认通过byName的方式实现,如果找不到名字,就通过byType实现。