Spring学习第三天:配置Bean-1

来源:互联网 发布:淘宝不搜血滴子 编辑:程序博客网 时间:2024/05/17 23:30

内容:

配置形式:基于XML文件的方式;基于注解的方式Bean的配置方法:通过全类名(反射),通过工厂方式(静态工厂方式&实例工厂方式),FactoryBeanIOC容器BeanFactory&ApplicationContext概述依赖注入的方式:属性注入,构造器注入注入属性值细节自动转配Bean之间的关系:继承,依赖Bean的作用域:singleton,prototype,WEB环境作用域使用外部属性文件spELIOC容器中Bean的生命周期Spring4.x新特性:泛型依赖注入

在Spring的IOC容器里配置Bean

  1. 在xml文件中通过bean节点来配置bean
<!-- 通过全类名的方式配置bean --><!-- class: bean的全类名,表示是通过反射的方式在IOC容器中创建bean实例。所以要求在HelloWorld中必须有一个无参的构造函数,如果不存在这个无参的构造器,那么就会编译出错 --><!-- id: 标志容器中的bean,id唯一 --><bean id="helloWorld" class="com.atguigu.spring.beans.HelloWorld" >    <property name="name" value="World"></property></bean>

id: bean的名称
- 在IOC容器中必须是唯一的
- 若id没有指定,Spring自动将类名作为Bean的名字
- id可以指定多个名字,名字之间可用逗号,分号,或空格分隔

Spring容器
在Spring IOC容器读取Bean配置创建Bean实例之前,必须对他进行实例化,只有在容器实例化之后,才可以从IOC容器里获取Bean实例并使用。
Spring提供了两种类型的IOC容器实现
- BeanFactory, IOC的基本实现
- ApplicationContext 提供了更多的高级特性,是BeanFactory的子接口
- BeanFactory是Spring框架的基础设施,面向Spring本身;ApplicationContext面向使用Spring框架的开发者,几乎所有的应用场合都直接使用ApplicationContext而非底层的BeanFactory
- 无论使用何种方式,配置文件是相同的

AplicationContext介绍

ApplicationContext类的UML图

ApplicationContext的主要实现类:
- ClassPathXmlApplicationContext, 从类路径下加载配置文件
- FileSystemXmlApplicationContext,从文件系统中加载配置文件

ConfigurableContext扩展于ApplicationContext,新增加两个主要方法:refresh()和close() ,让ApplicationContext具有启动,刷新和关闭上下文的能力。

ApplicationContext在初始化上下文时就实例化所有实例的Bean

WebApplicationContext是专门为WEB应用而准备的,它允许从相对于WEB根目录的路径中完成初始化工作

代码实例:

// 创建Spring 的IOC 容器// ApplicationContext 代表IOC容器// ClassPathXmlApplicationContext: 是ApplicationContext 接口的实现类,该实现类从类路径下来加载配置文件ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");//从IOC容器中获取bean实例//通过id 定位到IOC容器中的beanHelloWorld helloWorld = (HelloWorld) ctx.getBean("helloWorld2");//通过类型返回IOC容器中的bean,但要求IOC容器中必须只能有一个该类型的beanHelloWorld helloWorld2 = (HelloWorld) ctx.getBean(HelloWorld.class);

上述提到的,如果配置文件中有两个同一类型的bean,如下

<!-- 配置bean --><bean id="helloWorld" class="com.atguigu.spring.beans.HelloWorld" >    <property name="name" value="World"></property></bean><bean id="helloWorld2" class="com.atguigu.spring.beans.HelloWorld" >    <property name="name" value="World"></property></bean>

则会遇到如下错误

Exception in thread "main" org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type [com.atguigu.spring.beans.HelloWorld] is defined: expected single matching bean but found 2: helloWorld,helloWorld2    at org.springframework.beans.factory.support.DefaultListableBeanFactory.getBean(DefaultListableBeanFactory.java:312)    at org.springframework.context.support.AbstractApplicationContext.getBean(AbstractApplicationContext.java:985)    at com.atguigu.spring.beans.Main.main(Main.java:18)

从IOC容器中获取Bean的方式可以调用ApplicationContext的getBean()方法。
getBean方法

依赖注入的方式
Spring支持三种依赖注入的方式
- 属性注入
- 构造器注入
- 工厂方法注入(很少使用,不推荐)

属性注入
属性注入即通过setter方法注入Bean的属性值或依赖的对象
属性注入使用property元素,使用name属性指定Bean的属性名称,value属性或value子节点指定属性值
属性注入是实际应用中最常用的注入方式

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

Note: 因为属性注入是依赖于setter方法注入的,所以Bean类必须为变量提供setter方法。

构造器方式注入
通过构造方法注入Bean的属性值或依赖的对象,他保证了Bean实例在实例化后就可以使用
构造器注入在 constructor-arg元素中申明属性,constructor-arg 中没有name属性

新建一个Car类

public class Car {    private String brand;    private String corp;    private double price;    private int maxSpeed;    public Car(String brand, String corp, double price) {        super();        this.brand = brand;        this.corp = corp;        this.price = price;    }    @Override    public String toString() {        // TODO Auto-generated method stub        return brand + " " + corp + " " + price + " " + maxSpeed;    }   }

该类中使用构造器初始化Car类中的三个参数

xml文件配置:

<!-- 通过构造方法来配置Bean的属性 --><!-- 可以使用index下标确定那个属性 --><bean id="car" class="com.atguigu.spring.beans.Car">    <constructor-arg value="Audi"></constructor-arg>    <constructor-arg value="Shanghai" index="1"></constructor-arg>    <constructor-arg value="300000" index="2"></constructor-arg></bean>

调用这个Bean

Car car = ctx.getBean(Car.class);System.out.println(car);

输出如下结果:

Audi Shanghai 300000 0

上述的index可有可无,没有的话则按照构造器中参数的顺序自动注入。

如果再配置一个构造器:

public Car(String brand, String corp, int maxSpeed) {    super();    this.brand = brand;    this.corp = corp;    this.maxSpeed = maxSpeed;}

配置XML文件

<bean id="car2" class="com.atguigu.spring.beans.Car">    <constructor-arg value="baoma"></constructor-arg>    <constructor-arg value="Shanghai" index="1"></constructor-arg>    <constructor-arg value="240" index="2"></constructor-arg></bean>

调用Bean

Car car = (Car) ctx.getBean("car");System.out.println(car);car = (Car) ctx.getBean("car2");System.out.println(car);

输出结果

Audi Shanghai 300000.0 0baoma Shanghai 240.0 0

可见,并没有按照我们预想的,把240赋值给maxSpeed。

此时,仅仅依靠默认顺序,是无法识别重载的构造器的。 我们可以使用标签type去区别。

xml配置如下:

<bean id="car2" class="com.atguigu.spring.beans.Car">    <constructor-arg value="baoma"></constructor-arg>    <constructor-arg value="Shanghai"></constructor-arg>    <constructor-arg value="240" type="int"></constructor-arg>    <!-- 或者采用<value>的方法-->    <constructor-arg type="int">        <value>240</value>    </constructor-arg></bean>

运行结果如下:

Audi Shanghai 300000.0 0baoma Shanghai 0.0 240

从输出结果可知,使用构造器注入属性值的时候可以指定参数的位置和参数的类型,以区分重载构造器。
index 表示 构造器中的位置,0表示构造器第一个参数
type表示构造器中参数的类型。
这两者可以结合使用。
Note:与属性注入不同的是,使用构造器注入必须提供带参数的构造器。

0 0