内容提要
- IOC & DI 概述
- 配置 bean
- 配置形式:基于 XML 文件的方式;基于注解的方式
- Bean 的配置方式:通过全类名(反射)、通过工厂方法(静态工厂方法 & 实例工厂方法)、FactoryBean
- IOC 容器 BeanFactory & ApplicationContext 概述
- 依赖注入的方式:属性注入;构造器注入
- 注入属性值细节
- 自动转配
- bean 之间的关系:继承;依赖
- bean 的作用域:singleton;prototype;WEB 环境作用域
- 使用外部属性文件
- spEL
- IOC 容器中 Bean 的生命周期
- Spring 4.x 新特性:泛型依赖注入
IOC 和 DI
- IOC(Inversion of Control):其思想是反转资源获取的方向. 传统的资源查找方式要求组件向容器发起请求查找资源. 作为回应, 容器适时的返回资源. 而应用了 IOC 之后, 则是容器主动地将资源推送给它所管理的组件, 组件所要做的仅是选择一种合适的方式来接受资源. 这种行为也被称为查找的被动形式
- DI(Dependency Injection) — IOC 的另一种表述方式:即组件以一些预先定义好的方式(例如: setter 方法)接受来自如容器的资源注入. 相对于 IOC 而言,这种表述更直接
在 Spring 的 IOC 容器里配置 Bean
- 在 xml 文件中通过 bean 节点来配置 bean
- id:Bean 的名称。
-
- 在 IOC 容器中必须是唯一的
-
- 若 id 没有指定,Spring 自动将权限定性类名作为 Bean 的名字
-
- id 可以指定多个名字,名字之间可用逗号、分号、或空格分隔
<!-- 配置 bean class : bean 的全类名 , 通过反射的方式在 IOC 容器中创建 Bean . 所以要求 Bean 中必须有无参数的构造器
id : 标识容器中的bean . id 唯一 . -->
<bean id="helloWorld" class="com.HelloWorld">
<property name="name" value="Spring"></property>
</bean>
Spring容器
- 在 Spring IOC 容器读取 Bean 配置创建 Bean 实例之前, 必须对它进行实例化. 只有在容器实例化后, 才可以从 IOC 容器里获取 Bean 实例并使用.
- Spring 提供了两种类型的 IOC 容器实现.
-
- BeanFactory: IOC 容器的基本实现.
-
- ApplicationContext: 提供了更多的高级特性. 是 BeanFactory 的子接口.
-
- BeanFactory 是 Spring 框架的基础设施,面向 Spring 本身;ApplicationContext 面向使用 Spring 框架的开发者,几乎所有的应用场合都直接使用 ApplicationContext 而非底层的 BeanFactory
-
- 无论使用何种方式, 配置文件时相同的
ApplicationContext
- ApplicationContext 的主要实现类:
-
- ClassPathXmlApplicationContext:从类路径下加载配置文件
-
- FileSystemXmlApplicationContext: 从文件系统中加载配置文件
- ConfigurableApplicationContext 扩展于 ApplicationContext,新增加两个主要方法:refresh() 和 close(), 让 ApplicationContext 具有启动、刷新和关闭上下文的能力
- ApplicationContext 在初始化上下文时就实例化所有单例的 Bean。
- WebApplicationContext 是专门为 WEB 应用而准备的,它允许从相对于 WEB 根目录的路径中完成初始化工作
//1. 创建 Spring 的 IOC 容器对象.
//ApplicationContext 代表 IOC 容器
//ClassPathXmlApplicationContext : 是 ApplicationContext 接口的实现类,该实现类从类路径下加载配置文件
ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
//2. 从 IOC 容器中获取 Bean 实例.
//利用 ID 定位到 IOC 容器中的 bean
HelloWorld helloWorld1 = (helloworld)ctx.getBean("helloWorld");
//3.利用类型返回 IOC 容器中的 Bean , 但要求 IOC 容器中必须只能有一个该类型的 Bean
HelloWorld helloWorld2 = ctx.getBean(HelloWorld.class);
总结:使用IOC容器的getBean()方法获取返回的对象,可以使用在配置文件中设置的id获取对应的对象,当对象在IOC容器中只有一个时可以直接用类型返回IOC容器中的Bean.
依赖注入方式
Spring 支持 3 种依赖注入的方式
- 属性注入
- 构造器注入
- 工厂方法注入(很少使用,不推荐)
属性注入
- 属性注入即通过 setter 方法注入Bean 的属性值或依赖的对象
- 属性注入使用 property 元素, 使用 name 属性指定 Bean 的属性名称,value 属性或 value 子节点指定属性值
- 属性注入是实际应用中最常用的注入方式
<bean id="person" class="com.Person">
<property name="name" value="Tom"></property>
<property name="age" value="24"></property>
</bean>
构造方法注入
- 通过构造方法注入Bean 的属性值或依赖的对象,它保证了 Bean 实例在实例化后就可以使用。
- 构造器注入在 constructor-arg 元素里声明属性, constructor-arg 中没有 name 属性
<!-- 通过构造方法来配置 bean 的属性 -->
<bean id="car" class="aervice.Car">
<constructor-arg value="Audi" index="1"></constructor-arg>
<!-- 如果字面值包含特殊字符可以使用<![CDATA[]]> 包裹起来 -->
<!-- 属性值也可以使用value 直接进行配置 -->
<constructor-arg>
<value><![CDATA[<Shanghai>^]]></value>
</constructor-arg>
<constructor-arg value="3000000" type="double"></constructor-arg>
</bean>
<!-- 通过构造方法来配置 bean 的属性时可以指定参数的位置和参数的类型,以区分重载构造器 -->
<bean id="car2" class="aervice.Car">
<constructor-arg value="BWM"></constructor-arg>
<constructor-arg value="ShangHai"></constructor-arg>
<constructor-arg value="230" type="int"></constructor-arg>
</bean>
字面值
- 字面值:可用字符串表示的值,可以通过 < value> 元素标签或 value 属性进行注入。
- 基本数据类型及其封装类、String 等类型都可以采取字面值注入的方式
- 若字面值中包含特殊字符,可以使用< ![CDATA[]]> 把字面值包裹起来。
引用其他 Bean
- 组成应用程序的 Bean 经常需要相互协作以完成应用程序的功能. 要使 Bean 能够相互访问, 就必须在 Bean 配置文件中指定对 Bean 的引用
- 在 Bean 的配置文件中, 可以通过 < ref> 元素或 ref 属性为 Bean 的属性或构造器参数指定对 Bean 的引用.
- 也可以在属性或构造器里包含 Bean 的声明, 这样的 Bean 称为内部 Bean
<bean id="person" class="aervice.Person">
<property name="name" value="Tom"></property>
<property name="age" value="24"></property>
<!-- 可以使用 property 的 ref 属性建立 bean 之间的引用关系 -->
<!-- <property name="car" ref="car2"></property> -->
<!-- 内部 Bean , 不能被外部引用 , 只能在内部使用 -->
<property name="car">
<bean id="car3" class="aervice.Car">
<constructor-arg value="Ford"></constructor-arg>
<constructor-arg value="Changan"></constructor-arg>
<constructor-arg value="200000" type="double"></constructor-arg>
</bean>
</property>
</bean>
内部Bean
- 当 Bean 实例仅仅给一个特定的属性使用时, 可以将其声明为内部 Bean. 内部 Bean 声明直接包含在 < property> 或 < constructor-arg> 元素里, 不需要设置任何 id 或 name 属性
- 内部 Bean 不能使用在任何其他地方
<bean id="person" class="aervice.Person">
<property name="name" value="Tom"></property>
<property name="age" value="24"></property>
<!-- 可以使用 property 的 ref 属性建立 bean 之间的引用关系 -->
<!-- <property name="car" ref="car2"></property> -->
<!-- 内部 Bean , 不能被外部引用 , 只能在内部使用 -->
<property name="car">
<bean id="car3" class="aervice.Car">
<constructor-arg value="Ford"></constructor-arg>
<constructor-arg value="Changan"></constructor-arg>
<constructor-arg value="200000" type="double"></constructor-arg>
</bean>
</property>
</bean>
集合属性
- 在 Spring中可以通过一组内置的 xml 标签(例如: < list>, < set> 或 < map>) 来配置集合属性.
- 配置 java.util.List 类型的属性, 需要指定 < list> 标签, 在标签里包含一些元素. 这些标签可以通过 < value> 指定简单的常量值, 通过 < ref> 指定对其他 Bean 的引用. 通过< bean> 指定内置 Bean 定义. 通过 < null/> 指定空元素. 甚至可以内嵌其他集合.
- 数组的定义和 List 一样, 都使用 < list>
- 配置 java.util.Set 需要使用 < set> 标签, 定义元素的方法与 List 一样.
- Java.util.Map 通过 < map> 标签定义, < map> 标签里可以使用多个 < entry> 作为子标签. 每个条目包含一个键和一个值.
- 必须在 < key> 标签里定义键
- 因为键和值的类型没有限制, 所以可以自由地为它们指定 < value>, < ref>, < bean> 或 < null> 元素.
- 可以将 Map 的键和值作为 < entry> 的属性定义: 简单常量使用 key 和 value 来定义; Bean 引用通过 key-ref 和 value-ref 属性定义
- 使用 < props> 定义 java.util.Properties, 该标签使用多个 < prop> 作为子标签. 每个 < prop> 标签必须定义 key 属性.
<!-- 测试如何配置集合属性 -->
<bean id="person3" class="com.Spring.beans.collection.Person">
<property name="name" value="Mike"></property>
<property name="age" value="20"></property>
<property name="cars">
<!-- 使用 list 节点为 List 类型的属性赋值 -->
<list>
<ref bean="car" />
<ref bean="car2" />
</list>
</property>
</bean>
<!-- 配置 Map 属性值 -->
<bean id="newPerson" class="com.Spring.beans.collection.NewPerson">
<property name="name" value="Mit"></property>
<property name="age" value="29"></property>
<property name="cars">
<!-- 使用 Map 节点及map的entry子节点配置Map类型的成员变量 -->
<map>
<entry key="AA" value-ref="car"></entry>
<entry key="BB" value-ref="car2"></entry>
</map>
</property>
</bean>
<!-- 配置 Properties 属性值 -->
<bean id="dataSource"
class="com.Spring.beans.collection.DataSource">
<property name="properties">
<!-- 使用 props 和 prop 子节点来为 Properties 属性赋值 -->
<props>
<prop key="user">root</prop>
<prop key="password">123</prop>
<prop key="jdbcUrl">jdbc:mysql:///test</prop>
<prop key="driverClass">com.mysql.jdbc.Driver</prop>
</props>
</property>
</bean>
使用 utility scheme 定义集合
- 使用基本的集合标签定义集合时, 不能将集合作为独立的 Bean 定义, 导致其他 Bean 无法引用该集合, 所以无法在不同 Bean 之间共享集合.
- 可以使用 util schema 里的集合标签定义独立的集合 Bean. 需要注意的是, 必须在 < beans> 根元素里添加 util schema 定义
<!-- 配置独立的集合 bean , 以供多个 bean 进行引用 ,需要导入 util 命名空间-->
<util:list id="cars">
<ref bean="car"/>
<ref bean="car2"/>
</util:list>
<bean id = "person4" class="com.Spring.beans.collection.Person">
<property name="name" value="Jack"></property>
<property name="age" value="16"></property>
<property name="cars" ref="cars"></property>
</bean>
使用 p 命名空间
- 为了简化 XML 文件的配置,越来越多的 XML 文件采用属性而非子元素配置信息。
- Spring 从 2.5 版本开始引入了一个新的 p 命名空间,可以通过 < bean> 元素属性的方式配置 Bean 的属性。
- 使用 p 命名空间后,基于 XML 的配置方式将进一步简化
<!-- 通过 p 命名空间为 bean 的属性赋值, 需要先导入 p 命名空间 . 相对于传统的配置方式更加简洁-->
<bean id="person5" class="com.Spring.beans.collection.Person" p:age="30"
p:name="Queen" p:cars-ref="cars"
></bean>