配置Bean
反射配置
命名空间(p)
简化xml文件的配置
Spring从2.5版本开始引入了一个新的p命名空间,可以通过<bean>元素属性的方式配置bean的属性
集合属性
通过<map>标签定义,<map>标签里可以使用多个<entry>作为子标签,每个条目包含一个键和一个值
必须在<key>标签定义键
因为键和值的类型没有限制,所以可以自由地为他们制定<value>,<ref>,<bean>或<null>元素
可以将Map的键和值作为<entry>的属性定义:简单常量使用
key和value来定义;bean引用通过key-ref和value-ref属性定义
使用<props>定义java.util.Properties,该标签使用多个<prop>作为子标签,
每个<prop>标签必须定义key属性
使用utility scheme定义集合
使用基本的集合标签定义集合时,不能将集合作为独立的ben定义,
导致其他bean无法引用该集合,所以无法在不同bean之间共享集合
可以使用util schema里的集合标签定义独立的集合bean,需要注意的是必须在
<beans>根元素里添加util schema定义
自动装配(autowrie)
三种方式
Spring IOC容器可以自动装配bean需要做的仅仅是在
<bean>的autowire属性里指定自动装配的模式
byType(根据类型自动装配):若IOC容器中有多个与目标bean类型一致的bean,在这种情况下,
spring将无法判定哪个bean最合适该属性,所以不能执行自动装配
byName(根据名称自动装配):必须将目标bean的名称和属性名设置完全相同
constructor(通过构造器自动装配:当bean中存在多个构造器时,此种自动装配方式将会很复杂,不建议使用)
缺点
在 Bean 配置文件里设置 autowire 属性进行自动装配将会装配
Bean 的所有属性. 然而, 若只希望装配个别属性时,
autowire 属性就不够灵活了.
autowire 属性要么根据类型自动装配, 要么根据名称自动装配, 不能两者兼而有之.
一般情况下,在实际的项目中很少使用自动装配功能,因为和自动装配
功能所带来的好处比起来,明确清晰的配置文档更有说服力一些
工厂方法配置
静态
调用静态工厂方法创建 Bean是将对象创建的过程封装到静态方法中. 当客户端需要对象时, 只需要简单地调用静态方法, 而不同关心创建对象的细节.
要声明通过静态方法创建的bean,需要在bean的class属性里指定拥有该工厂的方法的类,同时在factory-method属性里指定工厂方法名称,最后,使用<constrctor-arg>元素为该方法传递方法参数
实例
将对象的创建过程封装到另外一个对象的实例方法里。当客户需要请求对象时,只需要简单的调用该实例方法而不需要关心对象的创建细节
要声明通过实例工厂方法创建的 Bean
在 bean 的 factory-bean 属性里指定拥有该工厂方法的 Bean
在 factory-method 属性里指定该工厂方法的名称
使用 construtor-arg 元素为工厂方法传递方法参数
FactoryBean
Spring中有两种类型的bean,一种是普通bean,另一种是工厂bean
工厂bean与普通bwan不同,其返回的对象不是指定类的一个实例,
其返回的是该工厂bean的getobject方法所返回的对象
注解配置
子主题
组件扫描(component scanning): Spring 能够从 classpath 下自动扫描, 侦测和实例化具有特定注解的组件.
特定组件包括:
@Component: 基本注解, 标识了一个受 Spring 管理的组件
@Respository: 标识持久层组件
@Service: 标识服务层(业务层)组件
@Controller: 标识表现层组件
对于扫描到的组件, Spring 有默认的命名策略: 使用非限定类名,
第一个字母小写. 也可以在注解中通过 value 属性值标识组件的名称
扫描组件
当在组件类上使用了特定的 注解之后,还需要在Spring的配置文件中声明<context:component:scan>
base-package属性指定一个需要扫描的基类包,Spring容器将会扫描这个基类包里及其子包中的所有类
当需要扫描多个包时,可以使用逗号分隔
如果仅希望扫描特定的类而非基包下的所有类,可使用resource_pattern属性过滤特定的类
<context:include-filter>子节点表示要包含的目标类
<context:exclude-filter>子节点表示要排除在外的目标类
<context:component-scan>下可以拥有若干个<context:include-filter>和<context:exclude_filter>子节点
组件装配
<context:component-scan> 元素还会自动注册 AutowiredAnnotationBeanPostProcessor 实例, 该实例可以自动装配具有 @Autowired 和 @Resource 、@Inject注解的属性.
@Autowired注解
@Autowired 注解自动装配具有兼容类型的单个 Bean属性
构造器, 普通字段(即使是非 public), 一切具有参数的方法都可以应用@Autowired 注解
默认情况下, 所有使用 @Autowired 注解的属性都需要被设置. 当 Spring 找不到匹配的 Bean 装配属性时, 会抛出异常, 若某一属性允许不被设置, 可以设置 @Authwired 注解的 required 属性为 false
默认情况下, 当 IOC 容器里存在多个类型兼容的 Bean 时, 通过类型的自动装配将无法工作. 此时可以在 @Qualifier 注解里提供 Bean 的名称. Spring 允许对方法的入参标注 @Qualifiter 已指定注入 Bean 的名称
@Authwired 注解也可以应用在数组类型的属性上, 此时 Spring 将会把所有匹配的 Bean 进行自动装配.
@Authwired 注解也可以应用在集合属性上, 此时 Spring 读取该集合的类型信息, 然后自动装配所有与之兼容的 Bean.
@Authwired 注解用在 java.util.Map 上时, 若该 Map 的键值为 String, 那么 Spring 将自动装配与之 Map 值类型兼容的 Bean, 此时 Bean 的名称作为键值
@Resource注解要求提供一个Bean名称的属性,若该属性为空,则自动采用标注的变量或方法名作为bean的名称
@Inject和@Autowired注解一样也是按类型匹配注入的Bean,但没有reqired属性
建议使用@Autowired注解
引用Bean
引用
组成应用程序的Bean经常需要相互协作以完成应用程序的功能。
要使Bean能够互相访问,就必须在Bean配置文件中指定对Bean的引用
在Bean的配置文件中,可以通过<ref>元素或ref属性为Bean的属性或构造参数指定对Bean的引用
当Bean实例仅仅给一个特定的属性使用时,可以将其声明为内部Bean,内部Bean声明直接包含在<property>或<constructor-arg>元素里,不需要设置任何id或name属性
内部bean不能被外部引用
bean直接的关系
继承
spring允许继承bean的配置,被继承的bean称为父bean,继承这个父bean的bean称为子bean
子bean从父bean中继承配置,包括bean的属性配置
子bean也可以覆盖父bean继承过来的配置
父bean可以作为配置模板,也可以作为bean实例,若只想把父bean作为模板,
可以设置<bean>的abstract属性为true,这样spring将不会实例化这个bean
并不是<bean>元素里所有属性都会被继承,比如:autowire,abstract等
也可以忽略父bean的class属性,让子bean指定自己的类,而共享相同的配置属性,但此时abstract必须设置为true
若一个bean的class属性没有指定,则该bean必须是一个抽象bean
依赖
Spring允许用户通过depends-on属性设定bean前置依赖的bean,前置依赖的bean会在本bean实例化之前创建好
如果前置依赖于多个bean,则可以通过逗号,空格的方式配置bean名称
bean作用域
在Spring中,可以在<bean>元素的scope属性里设置bean的作用域
默认情况下,Spring只为每个在IOC容器里声明的bean创建唯一的一个实例,整个IOC容器范围内部都能共享该实例。
所有后续的getBean调用和bean引用都将返回这个唯一的bean实例,该作用域被称为singleton,它是所有bean的默
认作用域
四种作用域
singleton
在SpringIOC容器中仅存在一个Bean实例,bean以单例的方式存在
prototype
每次调用getBean时都会返回一个新的实例
request
每次http请求都会创建一个新的bean,该作用域仅适用于WebpplicationContext环境
session
同一个HTTP Session共享一个Bean,不同的HTTP Sessionn使用不同的Beana.该作用域仅适用于WebApplicationContext环境
外部属性文件
在配置文件里配置 Bean 时, 有时需要在 Bean 的配置里混入系统部署的细节信息
(例如: 文件路径, 数据源配置信息等). 而这些部署细节实际上需要和 Bean 配置相分离
Spring 提供了一个 PropertyPlaceholderConfigurer 的 BeanFactory 后置处理器,
这个处理器允许用户将 Bean 配置的部分内容外移到属性文件中. 可以在 Bean 配置
文件里使用形式为 ${var} 的变量,
PropertyPlaceholderConfigurer 从属性文件里加载属性, 并使用这些属性来替换变量.
Spring 还允许在属性文件中使用 ${propName},以实现属性之间的相互引用。
Spring 2.5 之后: 可通过 <context:property-placeholder> 元素简化:
SpEL
简介
Spring 表达式语言(简称SpEL):是一个支持运行时查询和操作对象图的强大的表达式语言。
语法类似于 EL:SpEL 使用 #{…} 作为定界符,所有在大框号中的字符都将被认为是 SpEL
SpEL 为 bean 的属性进行动态赋值提供了便利
通过 SpEL 可以实现:
通过 bean 的 id 对 bean 进行引用
调用方法以及引用对象中的属性
计算表达式的值
正则表达式的匹配
字面量的表示
整数:<property name="count" value="#{5}"/>
小数:<property name="frequency" value="#{89.7}"/>
科学计数法:<property name="capacity" value="#{1e4}"/>
String可以使用单引号或者双引号作为字符串的定界符号:<property name=“name” value="#{'Chuck'}"/> 或 <property name='name' value='#{"Chuck"}'/>
Boolean:<property name="enabled" value="#{false}"/>
bean的生命周期
Spring IOC 容器可以管理 Bean 的生命周期, Spring 允许在 Bean 生命周期的特定点执行定制的任务.
pring IOC 容器对 Bean 的生命周期进行管理的过程:
通过构造器或工厂方法创建 Bean 实例
为 Bean 的属性设置值和对其他 Bean 的引用
调用 Bean 的初始化方法
Bean 可以使用了
当容器关闭时, 调用 Bean 的销毁方法
在 Bean 的声明里设置 init-method 和 destroy-method 属性, 为 Bean 指定初始化和销毁方法.
Bean 后置处理器允许在调用初始化方法前后对 Bean 进行额外的处理.
Bean 后置处理器对 IOC 容器里的所有 Bean 实例逐一处理, 而非单一实例.
其典型应用是: 检查 Bean 属性的正确性或根据特定的标准更改 Bean 的属性.
Spring IOC 容器对 Bean 的生命周期进行管理的过程:
通过构造器或工厂方法创建 Bean 实例
为 Bean 的属性设置值和对其他 Bean 的引用
将 Bean 实例传递给 Bean 后置处理器的 postProcessBeforeInitialization 方法
调用 Bean 的初始化方法
将 Bean 实例传递给 Bean 后置处理器的 postProcessAfterInitialization方法
Bean 可以使用了
当容器关闭时, 调用 Bean 的销毁方法