且构网

分享程序员开发的那些事...
且构网 - 分享程序员编程开发的那些事

Spring IoC — 基于XML的配置

更新时间:2022-10-04 21:09:46

原文:Spring IoC — 基于XML的配置

1、属性注入

注意点:
1)如果类中显示定义了一个带参的构造函数,则一定还要显示提供一个无参构造函数,否则使用属性注入时将抛出异常。
2)JavaBean关于属性命名的特殊规范。Spring只会检查Bean中是否有对应的Setter方法,至于Bean中是否有对应的属性变量则不做要求。如maxSpeed对应setMaxSpeed(),brand对应setBrand()。
所以<property>元素的属性变量名应满足:xxx的属性对应setXxx()方法。变量的前两个字母要么全部大写,要么全部小写。
 
Car类:
package com.ioc.ch4_3_1;
/**
 * Created by gao on 16-3-25.
 */
public class Car {
    private int maxSpeed;
    public String brand;
    private double price;
    public Car() {
    }
    public Car(int maxSpeed, String brand, double price) {
        this.maxSpeed = maxSpeed;
        this.brand = brand;
        this.price = price;
    }
    public void setMaxSpeed(int maxSpeed) {
        this.maxSpeed = maxSpeed;
    }
    public void setBrand(String brand) {
        this.brand = brand;
    }
    public void setPrice(double price) {
        this.price = price;
    }
    public int getMaxSpeed() {
        return maxSpeed;
    }
    public String getBrand() {
        return brand;
    }
    public double getPrice() {
        return price;
    }
    public String toString() {
        return "brand:" + brand + "/maxSpeed:" + maxSpeed + "/price:" + price;
    }
}

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
         http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">
    <bean id="car" class="com.ioc.ch4_3_1.Car">
        <property name="maxSpeed"><value>300</value></property>
        <property name="brand"><value>奥迪</value></property>
        <property name="price"><value>150000.00</value></property>
    </bean>
</beans>

测试类:

package com.ioc.ch4_3_1;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
/**
 * Created by gao on 16-3-25.
 */
public class Test {
    public static void main(String[] args) {
        ApplicationContext ctx = new ClassPathXmlApplicationContext("classpath:com\\ioc\\ch4_3_1\\beans.xml");
        Car car = (Car) ctx.getBean("car");
        System.out.println(car.toString());
    }
}
输出结果:
brand:奥迪/maxSpeed:300/price:150000.0
 
2、构造函数注入
注意点:循环依赖问题。Spring容器能顺利实例化以构造函数注入方式配置的Bean有一个前提是:Bean构造函数入参引用的对象必须已经准备就绪。如果两个Bean都采用构造函数注入,而且都通过构造函数入参引用对方,则会发生循环依赖问题。这时可以将构造函数注入方式调整为属性注入方式。
Car类:
package com.ioc.ch4_3_2;
/**
 * Created by gao on 16-3-25.
 */
public class Car {
    private int maxSpeed;
    public String brand;
    private double price;
    private String corp;
    public Car() {
    }
    public Car(String brand, double price) {
        this.brand = brand;
        this.price = price;
    }
    public Car(int maxSpeed, String brand, double price) {
        this.maxSpeed = maxSpeed;
        this.brand = brand;
        this.price = price;
    }
    public Car(String brand, String corp, int maxSpeed) {
        this.brand = brand;
        this.corp = corp;
        this.maxSpeed = maxSpeed;
    }
    public void setMaxSpeed(int maxSpeed) {
        this.maxSpeed = maxSpeed;
    }
    public void setBrand(String brand) {
        this.brand = brand;
    }
    public void setPrice(double price) {
        this.price = price;
    }
    public String getCorp() {
        return corp;
    }
    public void setCorp(String corp) {
        this.corp = corp;
    }
    public int getMaxSpeed() {
        return maxSpeed;
    }
    public String getBrand() {
        return brand;
    }
    public double getPrice() {
        return price;
    }
    public String toString() {
        return "brand:" + brand + "\tmaxSpeed:" + maxSpeed + "\tprice:" + price + "\tcorp:" + corp;
    }
}

Boss类:

package com.ioc.ch4_3_2;
public class Boss {
    private String name;
    private Car car;
    private Office office;
    public Boss(String name, Car car, Office office) {
        this.name = name;
        this.car = car;
        this.office = office;
    }
    public Boss(String name, Car car) {
        this.name = name;
        this.car = car;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public Car getCar() {
        return car;
    }
    public void setCar(Car car) {
        this.car = car;
    }
    public String toString(){
        return "name:"+name+"\tcar:"+car.getBrand()+"\toffice:"+office;
    }
}

Office类:

package com.ioc.ch4_3_2;
public class Office {
   
}

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
         http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">
    <!--构造函数注入:type -->
    <bean id="car1" class="com.ioc.ch4_3_2.Car">
        <constructor-arg type="java.lang.String">
            <value>car1_甲壳虫</value>
        </constructor-arg>
        <constructor-arg type="double">
            <value>300000.0</value>
        </constructor-arg>
    </bean>
    <!--构造函数注入:index-->
    <bean id="car2" class="com.ioc.ch4_3_2.Car">
        <constructor-arg index="0" value="300"/>
        <constructor-arg index="1" value="car2_宝马"/>
        <constructor-arg index="2" value="200000.0"/>
    </bean>
    <!--构造函数注入:type&index -->
    <bean id="car3" class="com.ioc.ch4_3_2.Car">
        <constructor-arg index="0" type="java.lang.String">
            <value>car3_红旗CA72</value>
        </constructor-arg>
        <constructor-arg index="1" type="java.lang.String">
            <value>中国一汽</value>
        </constructor-arg>
        <constructor-arg index="2" type="int">
            <value>200</value>
        </constructor-arg>
    </bean>
    <!--构造函数注入:自动识别入参类型 -->
    <bean id="boss1" class="com.ioc.ch4_3_2.Boss">
        <constructor-arg>
            <value>John</value>
        </constructor-arg>
        <constructor-arg>
            <ref bean="car1" />
        </constructor-arg>
        <constructor-arg>
            <ref bean="office" />
        </constructor-arg>
    </bean>
    <bean id="car" class="com.ioc.ch4_3_2.Car"/>
    <bean id="office" class="com.ioc.ch4_3_2.Office" />
 </beans>

测试类:

package com.ioc.ch4_3_2;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
/**
 * Created by gao on 16-3-25.
 */
public class Test {
    public static void main(String[] args) {
        ApplicationContext ctx = new ClassPathXmlApplicationContext("classpath:com\\ioc\\ch4_3_2\\beans.xml");
        Car car1 = (Car) ctx.getBean("car1");
        System.out.println(car1.toString());
        System.out.println("--------------------------------");
        Car car2 = (Car) ctx.getBean("car2");
        System.out.println(car2.toString());
        System.out.println("--------------------------------");
        Car car3 = (Car) ctx.getBean("car3");
        System.out.println(car3.toString());
        System.out.println("--------------------------------");
        Boss boss1 = (Boss) ctx.getBean("boss1");
        System.out.println(boss1.toString());
    }
}
输出结果:
brand:car1_甲壳虫 maxSpeed:0 price:300000.0 corp:null
--------------------------------
brand:car2_宝马 maxSpeed:300 price:200000.0 corp:null
--------------------------------
brand:car3_红旗CA72 maxSpeed:200 price:0.0 corp:中国一汽
--------------------------------
name:John car:car1_甲壳虫 office:com.ioc.ch4_3_2.Office@9eed10a
 
3、工厂方法注入
有非静态工厂方法和静态工厂方法两种方式
Car类:
package com.ioc.ch4_3_3;
/**
 * Created by gao on 16-3-25.
 */
public class Car {
    private int maxSpeed;
    public String brand;
    private double price;
    public Car() {
    }
    public Car(int maxSpeed, String brand, double price) {
        this.maxSpeed = maxSpeed;
        this.brand = brand;
        this.price = price;
    }
    public void setMaxSpeed(int maxSpeed) {
        this.maxSpeed = maxSpeed;
    }
    public void setBrand(String brand) {
        this.brand = brand;
    }
    public void setPrice(double price) {
        this.price = price;
    }
    public int getMaxSpeed() {
        return maxSpeed;
    }
    public String getBrand() {
        return brand;
    }
    public double getPrice() {
        return price;
    }
    public String toString() {
        return "brand:" + brand + "\tmaxSpeed:" + maxSpeed + "\tprice:" + price;
    }
}

CarFactory类:

package com.ioc.ch4_3_3;
public class CarFactory {
    //创建Car的工厂方法
   public Car createHongQiCar(){
       Car car = new Car();
       car.setBrand("car5_奔驰");
       return car;
   }
   //工厂类方法是静态的
   public static Car createCar(){
       Car car = new Car();
       return car;
   }
}

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
         http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">
    <bean id="carFactory" class="com.ioc.ch4_3_3.CarFactory"/>
    <bean id="car5" factory-bean="carFactory" factory-method="createHongQiCar"/>
    <bean id="car6" class="com.ioc.ch4_3_3.CarFactory" factory-method="createCar"/>
 </beans>

测试类:

package com.ioc.ch4_3_3;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
/**
 * Created by gao on 16-3-25.
 */
public class Test {
    public static void main(String[] args) {
        ApplicationContext ctx = new ClassPathXmlApplicationContext("classpath:com\\ioc\\ch4_3_3\\beans.xml");
        Car car5 = (Car) ctx.getBean("car5");
        System.out.println(car5.toString());
        System.out.println("-----------------------");
        Car car6 = (Car) ctx.getBean("car6");
        System.out.println(car6.toString());
    }
}
输出结果:
brand:car5_奔驰 maxSpeed:0 price:0.0
-----------------------
brand:null maxSpeed:0 price:0.0

 

注入方法的考量:
1)支持使用构造函数注入的理由:
    · 构造函数可以保证一些重要的属性在Bean实例化时就设置好,避免了因为一些重要属性没有提供,导致一个无用Bean实例的情况;
    · 不需要为每个属性提供Setter方法,减少了类的方法个数;    
    · 可以更好地封装类变量,不需要为每个属性指定setter方法,避免外部错误的调用。
2)更多的开发者更倾向于使用属性注入方式,反对构造函数注入的理由:
    · 如果一个类的属性众多,构造函数的签名将变成一个庞然大物,可读性很差;
    · 灵活性不强,在有些属性是可选的情况下,如果通过构造函数注入,也需要为可选的参数提供一个null值;
    · 如果有多个构造函数,需要考虑配置文件和具体构造函数匹配歧义的问题,配置上相对复杂;
    · 构造函数不利于类的继承和扩展,因为子类需要引用到父类复杂的构造函数;
    · 构造函数注入有时会造成循环依赖的问题。
3)对于一个全新的应用来说,不推荐使用工厂方法的注入方式。