Spring之IOC自动装配解析

         林炳文Evankaka原创作品。转自http://blog.csdn.net/evankaka

     set注入和构造注入有时在做配置时比较麻烦。所以框架为了提高开发效率,提供自动装配功能,简化配置。Spring框架式默认不支持自动装配的,要想使用自动装配需要修改spring配置文件中<bean>标签的autowire属性。自动装配属性有5个值可选,分别代表不同的含义。

本文工程免费下载

1、byName

 从Spring环境中获取目标对象时,目标对象中的属性会根据名称在整个Spring环境中查找<bean>标签的id属性值。如果有相同的,那么获取这个对象,实现关联。

 整个Spring环境:表示所有的spring配置文件中查找,那么id不能有重复的。

 

2、byType

 从Spring环境中获取目标对象时,目标对象中的属性会根据类型在整个spring环境中查找<bean>标签的class属性值。如果有相同的,那么获取这个对象,实现关联。

        缺点:如果存在多个相同类型的bean对象,会出错。

如果属性为单一类型的数据,那么查找到多个关联对象会发生错误。

如果属性为数组或集合(泛型)类型,那么查找到多个关联对象不会发生异常。

3、constructor(3.x以上已不能用)

 使用构造方法完成对象注入,其实也是根据构造方法的参数类型进行对象查找,相当于采用byType的方式。如果没找到则抛出异常

4、autodetect

 自动选择:如果对象没有无参数的构造方法,那么自动选择constructor的自动装配方式进行构造注入。如果对象含有无参数的构造方法,那么自动选择byType的自动装配方式进行setter注入。                      

5、no

默认情况下,不自动装配,通过“ref”attribute手动设定。

        <bean>标签的 autowire 属性,它负责自动装配<bean>标签定义 JavaBean 的属性。这样做可以省去很多配置 JavaBean 属性的标签代码,使代码整洁、美观。但是它也有负面影响,即使用自动装配之后,无法从配置文件中读懂 JavaBean 需要什么属性。自动装配存在很多不正确的装配问题,例如错误装载属性、“byType”属性和“constructor”属性对相同类型参数无法判断等。当然,将自动装配和手动装配混合使用也能解决此问题。下面通过一个实例来分析如何使用自动装配。

首先创建创建一个学生类 Student,定义学号、姓名、性别、年龄等属性,并添加对

应的 set()与 get()方法。程序代码如下。

  1. package com.autobean;
  2. public class Student {
  3. private String ID;
  4. private String name;
  5. private int age;
  6. private String sex;
  7. public String getID() {
  8. return ID;
  9. }
  10. public void setID(String iD) {
  11. ID = iD;
  12. }
  13. public String getName() {
  14. return name;
  15. }
  16. public void setName(String name) {
  17. this.name = name;
  18. }
  19. public int getAge() {
  20. return age;
  21. }
  22. public void setAge(int age) {
  23. this.age = age;
  24. }
  25. public String getSex() {
  26. return sex;
  27. }
  28. public void setSex(String sex) {
  29. this.sex = sex;
  30. }
  31. }
同样创建一个教师类 Teacher,定义姓名、性别和年龄等属性,并添加对应的 set()与 get()方法。程序代码如下。
  1. package com.autobean;
  2. public class Teacher {
  3. private String name;
  4. private int age;
  5. private String sex;
  6. public String getName() {
  7. return name;
  8. }
  9. public void setName(String name) {
  10. this.name = name;
  11. }
  12. public int getAge() {
  13. return age;
  14. }
  15. public void setAge(int age) {
  16. this.age = age;
  17. }
  18. public String getSex() {
  19. return sex;
  20. }
  21. public void setSex(String sex) {
  22. this.sex = sex;
  23. }
  24. }

创建教学档案类 TeachFile,定义 Teacher 和 Student 两个属性,并添加 print()方法。用于输出教师与学生的信息。程序代码如下。 

  1. package com.autobean;
  2. public class TeachFile {
  3. private Teacher teacher;
  4. private Student student;
  5. public TeachFile() {
  6. }
  7. public TeachFile(Teacher teacher, Student student) {
  8. this.teacher = teacher;
  9. this.student = student;
  10. }
  11. public Student getStudent() {
  12. return this.student;
  13. }
  14. public void setStudent(Student student1) {
  15. this.student = student1;
  16. }
  17. public Teacher getTeacher() {
  18. return teacher;
  19. }
  20. public void setTeacher(Teacher teacher) {
  21. this.teacher = teacher;
  22. }
  23. public void print() {
  24. System.out.println("------教师信息------");
  25. System.out.println("姓名:" + teacher.getName());
  26. System.out.println("年龄:" + teacher.getAge());
  27. System.out.println("性别:" + teacher.getSex());
  28. System.out.println();
  29. System.out.println("------学生信息------");
  30. System.out.println("学号:" + student.getID());
  31. System.out.println("姓名:" + student.getName());
  32. System.out.println("年龄:" + student.getAge());
  33. System.out.println("性别:" + student.getSex());
  34. }
  35. }

在配置文件applicationContext.xml中定义刚刚创建的类,并为其赋值。其中 TeachFile 类采用了自动装配。程序代码如下。

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p"
  4. xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.2.xsd">
  5. <bean id="student" class="com.autobean.Student">
  6. <property name="ID" value="80" />
  7. <property name="name" value="阿王" />
  8. <property name="age" value="23" />
  9. <property name="sex" value="男" />
  10. </bean>
  11. <bean id="teacher" class="com.autobean.Teacher">
  12. <property name="name" value="何老师" />
  13. <property name="age" value="43" />
  14. <property name="sex" value="女" />
  15. </bean>
  16. <!-- 默认情况下,通过'ref’来装配bean -->
  17. <bean id="teachFile1" class="com.autobean.TeachFile" >
  18. <property name="teacher" ref="teacher" />
  19. <property name="student" ref="student" />
  20. </bean>
  21. <!--根据byName自动装配bean -->
  22. <bean id="teachFile2" autowire="byName" class="com.autobean.TeachFile" />
  23. <!--根据byType自动装配bean -->
  24. <bean id="teachFile3" autowire="byType" class="com.autobean.TeachFile" />
  25. <!--根据constructor自动装配bean -->
  26. <bean id="teachFile4" autowire="constructor" class="com.autobean.TeachFile"/>
  27. </beans>

在这个配置文件中定义了 Student 类和 Teacher 类,并为姓名、年龄和性别属性赋值。在定义 TeachFile 类时,没有传递任何参数,而是采用了 autowire 属性自动配置 TeachFile类所需要的属性。下面编写一个主类 PrintInfo 类来输出档案信息。程序代码如下。

  1. package com.autobean;
  2. import org.springframework.beans.factory.BeanFactory;
  3. import org.springframework.beans.factory.xml.XmlBeanFactory;
  4. import org.springframework.core.io.ClassPathResource;
  5. import org.springframework.core.io.Resource;
  6. public class PrintInfo {
  7. public static void main(String[] args) {
  8. Resource res = new ClassPathResource("applicationContext.xml");
  9. BeanFactory bf = new XmlBeanFactory(res);
  10. TeachFile tf1 = (TeachFile) bf.getBean("teachFile1");
  11. TeachFile tf2 = (TeachFile) bf.getBean("teachFile2");
  12. TeachFile tf3 = (TeachFile) bf.getBean("teachFile3");
  13. TeachFile tf4 = (TeachFile) bf.getBean("teachFile4");
  14. System.out.println("默认情况下,通过'ref’来装配bean");
  15. tf1.print();
  16. System.out.println("根据byName自动装配bean");
  17. tf2.print();
  18. System.out.println("根据byType自动装配bean");
  19. tf3.print();
  20. System.out.println("根据constructor自动装配bean");
  21. tf4.print();
  22. }
  23. }

输出结果:

四月 02, 2015 8:16:48 下午 org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions
信息: Loading XML bean definitions from class path resource [applicationContext.xml]
默认情况下,通过'ref’来装配bean
------教师信息------
姓名:何老师
年龄:43
性别:女

------学生信息------
学号:80
姓名:阿王
年龄:23
性别:男
根据byName自动装配bean
------教师信息------
姓名:何老师
年龄:43
性别:女

------学生信息------
学号:80
姓名:阿王
年龄:23
性别:男
根据byType自动装配bean
------教师信息------
姓名:何老师
年龄:43
性别:女

------学生信息------
学号:80
姓名:阿王
年龄:23
性别:男
根据constructor自动装配bean
------教师信息------
姓名:何老师
年龄:43
性别:女

------学生信息------
学号:80
姓名:阿王
年龄:23
性别:男


在上面实例的配置文件 appContext.xml 中,TeachFile 类采用4种装配,将 Teacher
类和 Student 类注入到对应的属性中。语法格式如下。

<bean autowire="byName" id="teachFile" class="TeachFile" />
在 autowire 属性中指定类型为“byName”。autowire 属性共支持 5 种装配类型,下面
分别介绍每种装配类型的用法。
(1)no:autowire 采用的默认值,采用自动装配。必须使用 ref 直接引用其他 Bean,这样可以增加代码的可读性,并且不易出错。
(2)byName:以属性名区分自动装配。在容器中寻找与 JavaBean 的属性名相同的JavaBean,并将其自动装配到 JavaBean 中。如果用上面的实例来解释,TeachFile 类的实例对象 teachFile 包含的两个属性分别是 Teacher 类和 Student 类的实例对象,而配置文件中已经定义了这两个类的实例。在定义 teachFile 实例时指定了自动装配类型为“byName”,容器会自动寻找 teachFile 实例需要的属性(即 teacher 和 student 两个 JavaBean),并注入到 teachFile 实例中。此类自动装配类型存在错误装配 JavaBean 的可能,如果配置文件中定义了与需要自动装配的 JavaBean 属性相同而类型不同的 JavaBean,那么它会错误地注入不同类型的JavaBean。读者可以将上面实例中的配置文件修改一下,将 student 和 teacher 两个JavaBean 的类型保持不变,将名字调换一下,便会出现此问题。这时自动装配无法解决此问题,只能通过混合使用手动装配来指定装配哪个 JavaBean。
(3)byType:以属性类型区分自动装配。容器会自动寻找与 JavaBean 的属性类型相同的 JavaBean 的定义,并将其注入到需要自动装配的 JavaBean 中。如果将上面配置JavaBean 自动装配的类型修改为 byType,也可以实现相同的结果。这种自动装配类型也会出现无法自动装配的情况。例如在配置文件中再次添加一个Student 类或 Teacher 类的实现对象,byType 自动装配类型会因为无法自动识别装配哪一个 JavaBean 而抛出org.springframework.beans.factory.UnsatisfiedDependencyException 异常。要解决此
问题,只能通过混合使用手动装配来指定装配哪个 JavaBean。
(4)constructor:通过构造方法的参数类型自动装配。此类型会使容器自动寻找与JavaBean 的构造方法的参数类型相同的 Bean,并注入到需要自动装配的 JavaBean 中。它
与 byType 类型存在相同的无法识别自动装配的情况。
(5)autudetect:这是最后一个自动装配类型,它首先使用 constructor 方式自动装配,然后使用 byType 方式。当然它也存在与 byType 和 constructor 相同的异常情况。建

议在使用自动装配时,把容易出现问题的 JavaBean 使用手动装配注入依赖属性

本文工程免费下载

         林炳文Evankaka原创作品。转自http://blog.csdn.net/evankaka

上一篇:持久化类和持久化对象的区别


下一篇:10 Java面向对象之封装