spring4之依赖注入的三种方式

时间:2021-05-07 06:50:58

1、Setter注入

        <bean id="helloWorld" class="com.jdw.spring.beans.HelloWorld">
<property name="name" value="Spring"></property>
</bean>

  从这个最简单的配置可以得出以下两个结论:一、class属性需要全类名进行配置,说了ioc容器是通过反射进行创建bean的;二、bean类必须有默认的构造函数;

  这种通过property子元素进行属性配置的方式是spring依赖注入的三大方式之一:setter注入。

2、构造器注入

         <bean id="car" class="com.jdw.spring.beans.Car">
<constructor-arg value="audi" type="java.lang.String"></constructor-arg>
<constructor-arg type="java.lang.String">
<!-- 对于包含特殊字符的属性值,可以在value子节点使用CDATA -->
<value><![CDATA[<shanghai>]]></value>
</constructor-arg>
<constructor-arg type="int">
<value>240</value>
</constructor-arg>
</bean>

  构造器注入,言外之意,需要在bean之中配置构造器,可以不需要默认构造方法,但是必须保证构造器参数与配置文件中的参数数量和类型一致。

	<bean id="car1" class="com.jdw.spring.beans.Car">
<constructor-arg value="audi" index="0" type="java.lang.String"></constructor-arg>
<constructor-arg value="shanghai" index="1" type="java.lang.String"></constructor-arg>
<constructor-arg value="200000" index="2" type="double"></constructor-arg>
</bean>

  对于有多个构造器的bean,可通过上述的index和type属性进行配置,以区别重载的构造器。

3、工厂方法注入

    工厂方法注入又分为静态工厂方法注入和实例工厂方法注入

(1)静态工厂方法注入

一个模拟的简单静态工厂如下,该静态工厂提供了一个静态方法getCar,调用该方法则返回一个Car实例,注意红色注释,如果配置文件中scope=singleton,采用静态工厂方法注入获得的bean是否为单例呢?

public class StaticCarFactory {

	public static Map<String,Car> cars=new HashMap<>();
static{
cars.put("audi", new Car("audi",300000));
cars.put("bmw", new Car("bmw",200000));
cars.put("hongqi", new Car("hongqi",600000));
}
public static Car getCar(String name){
//return new Car("audi",300000);//如果这么写,那么获得的car实例还是单例的么?
return cars.get(name);
}
}

  静态工厂注入配置:

        <bean id="car" class="com.jdw.spring.factory.StaticCarFactory"
factory-method="getCar">
<constructor-arg value="audi"></constructor-arg>
</bean>

说明:所谓静态工厂方法注入,不过是通过一个静态方法创建bean的实例,注意,在singelton情况下该静态方法只会在ioc容器初始化bean时调用一次,即不管getBean("car")多少次,都是返回的同一个Car对象。而在prototype情况下,getBean("car")调用一次就会对应的调用一次静态方法getCar。从这里可以更好的理解singleton和prototype的区别,前者会在ioc容器初始化时创建,后者则在用户调用时再创建。

(2)实例工厂方法注入

  实例工厂代码:

public class InstanceCarFactory {
private Map<String, Car> cars = null; public InstanceCarFactory() {
cars = new HashMap<>();
cars.put("audi", new Car("audi", 30000));
cars.put("BMW", new Car("audi", 20000));
cars.put("HQ", new Car("audi", 50000));
} public Car getCar(String brand) {
return cars.get(brand);
}
}

  实例工厂的getCar为非静态,这就导致,必须先实例化该工厂,即需要配置该工厂的bean。

        <!-- 配置实例工厂 -->
<bean id="carFactory" class="com.jdw.spring.factory.InstanceCarFactory"></bean> <!-- 通过实例工厂配置bean -->
<bean id="car1" factory-bean="carFactory" factory-method="getCar">
<constructor-arg name="brand" value="audi"></constructor-arg>
</bean>

  学而不思则罔,通过Setter注入和构造器注入可以很方便快捷的配置bean,静态工厂注入和实例工厂注入到底是为了那样?难道只是为了支持下工厂模式?spring你别逗好么,咱自己不就相当于一个超级bean工厂么!那干嘛还弄给工厂方法注入呢?一定有其原因。

有一家手机卖场,只卖A、B、C三种手机,假设顾客只关心价格price和参数信息info,那么问题来了,一个顾客进入卖场,说:我要买A手机。手机卖场(IOC)需要初始化一个A的Bean。又一个顾客进入卖场,说:我要B手机。手机卖场也得初始化一个B的bean。同样,也得初始化C的bean。三个手机还好些,IOC就建立三个bean,那么10种手机呢?每种手机的info需要特殊处理呢?比如手机参数信息要有一系列的运算统计信息呢?这样看来IOC貌似不适合把每个bean都放在配置文件里,也不适合建立多个bean。

  解决方案就是,建立一个手机bean,通过工厂注入(静态工厂、实例工厂都可以),传入手机类型名称,获得对应的手机,只是要把bean配置为prototype,这样运行时就可以获取各式各样的手机了!

那静态工厂和实例工厂两种注入方式,除了配置的不同,在应用上有什么区别么?