java

位置:IT落伍者 >> java >> 浏览文章

Spring创建Bean的过程


发布日期:2019年11月21日
 
Spring创建Bean的过程

beans包提供了以编程方式管理和操作bean的基本功能而context包增加了ApplicationContext它以一种更加面向框架的方式增强了BeanFactory的功能

context包的基础是位于orgntext包中的ApplicationContext接口它是由BeanFactory接口派生而来的提供了BeanFactory的所有功能为了以一种更面向框架的方式工作使用分层和继承关系的上下文context包还提供了一下的功能

a MessageSource对IN消息的访问

b 资源访问例如URL和文件

c 事件传递给是吸纳了ApplicationListener接口的bean

d 载入多个(有继承关系)上下文使得每一个上下文都专注于一个特定的层是比如应用的web层

国际化支持

ApplicationContext扩展了MessageSource接口因而提供了messaging的功能(IN或者国际化)同NestingMessageSource一起使用还能处理分级的信息这些是spring提供的处理信息的基本接口

当一个ApplicationContext被加载时它会自动查找在context中定义的MessageSource bean这个bean必须交做messageSource如果找了这样一个bean所有对上述方法的调用将被委托给找到的messageSource如果没有找到messageSourceApplicationContext将会尝试查找他的父亲是否包含有同名的bean如果有它将把找到的bean作为MessageSource如果他没有找到任何的信息处理源他会创建一个StaticMessageSource

Spring目前提供了两个MessageSource的实现他们是

ResourceBundleMessageSource和StaticMessageSource两者都实现了NestingMessageSource一边能够处理嵌套的信息StaticMessageSource很少被使用但是他提供了编程的方式向source增加信息我们经常使用的是ResourceBundleMessageSource

在Spring中使用资源

ApplicationContext继承了ResourceLoader接口在这个接口中定义了getResource()方法如下

Resource getResource(String location)

该方法返回一个资源句柄这个句柄应该总是一个可重复使用的资源描述符允许多次调用getInputStream();

getResource()方法的参数是一个资源访问地址例如

file:\c:/testdata

classpath:testdata(从classpath路径中查找testdat文件并返回他的资源句柄)

WEBINF/testdat

注意getResource()方法返回的Resource句柄并不意味着资源实际存在你需要调用Resource接口的exists()方法判断资源是否存在

Resource提供了与协议无关的特性

事件传递

ApplicationContext中的时间处理是通过AppcationEvent类和ApplicationListener接口来提供的如果上下文中部署了一个实现了ApplicationListener接口的bean每次一个ApplicationEvent发布到ApplicationContext时那个bean就会被通知实质撒谎功能这是标准的Observer设计模式

Spring提供的三个标准事件

a ContextRefreshedEvent

当ApplicationContext已经初始化或刷新后发送的事件这里初始化意味着所有的bean被装载singleton被预实例化以及ApplicationContext已经准备好

b ContextClosedEvent

当使用ApplicationContext的close()方法结束上下文的时候发送的事件这里意味着singleton被销毁

c RequestHandledEvent

一个与web相关的事件告诉所有的bean一个HTTP请求已经被响应了(这个时间将会在一个请求结束后被发送—)注意这个时间只能应用于使用了Spring的DispatcherServlet的web应用

=========================================================================================

LifeCycle

InitializingBean/initmethod

实现orgspringframeworkbeansfactoryInitializingBean接口允许一个bean在他的所有必需的属性被BeanFactory设置后来执行初始化的工作

当然可以使用initmethod来取代实现这个接口以让应用不与spring产生耦合

如果一个bean既实现了InitializingBean又指定了initmethod则spring会先调InitializingBean的方法在调initmethod指定的方法

DisposableBean/destroymethod

实现orgspringframeworkbeansfactoryDisposableBean接口允许一个bean可以在包含他的BeanFactory销毁的时候得到一个回调

注意BeanFactory对bean的管理默认是单实例的如果bean不是单示例的spring就不能管理他的生命周期

BeanFactoryAware

对于实现了orgspringframeworkbeansfactoryBeanFactoryAware接口的类当它被BeanFactory创建后它会拥有一个指向创建他的BeanFactory的引用

BeanNameAware

如果一个bean实现了orgspringframeworkbeansfactoryBeanNameAware接口并且被部署到一个BeanFactory中那么BeanFactory就会通过这个接口来调用bean以便通知这个bean他被部署的id这个回调发生在普通的bean属性设置之后在初始化回调之前比如InitializingBean的afterProperteis方法(或者自定义的initmethod)

Bean的生命周期如下

Bean的构造

调用setXXX()方法设置Bean的属性

调用BeanNameAware的setBeanName();

调用BeanFactoryAware的setBeanFactory()方法

调用BeanPostProcessor的postProcessBeforeInitialization()方法

调用InitializingBean的afterPropertiesSet()方法

调用自定义的初始化方法

调用BeanPostProcessor类的postProcessAfterInitialization()方法

调用DisposableBean的destroy()方法

调用自定义的销毁方法

========================================================================================

扩展Spring的Ioc框架

Spring框架的IoC组件被设计为可扩展的通常应用开发者并不需要子类化各个BeanFactory或ApplicationContext的实现类通过插入特定接入接口的实现Spring的IoC容器就可以不受限制的进行扩展

BeanPostProcessor:在创建bean之后调用

BeanFactoryPostProcessors在创建bean之前调用

如果想在spring容器完成一个bean的实例化后再对他进行初始化之前或之后执行一些自定义的逻辑可以插入一个或多个BeanPostProcessor的实例

orgspringframewonfigBeanPostProcessor接口包含了两个回调方法当一个类作为容器的后置处理器(postprocessor)被注册后对于由容器创建的每个bean实例在任一个初始化方法(例如afterProperties和利用initmethod声明的方法)调用前后后置处理器会从容器中分别获取一个回调后置处理器可以随意对这个bean实例执行他所期望的动作也包括完全忽略这个回调

BeanFactory和ApplicationContext对待bean后置处理器稍有不同

ApplicationContext会自动检测任何提供给他的在配置元数据中定义实现了BeanPostProcessor接口的bean并把它们注册为后置处理器然后在容器创建bean的适当时候调用它部署一个后置处理器同部署其他的bean并没有什么区别无需其他的动作而另一方面当使用BeanFactory的时候bean后置处理器鼻息编写代码显示的去注册

我们看到的下一个扩展点是

orgspringframewonfigBeanFactoryPostProcessor出一个主要的区别外这个接口的寓意类似于BeanPostProcessorBeanFactoryPostProcessor作用于bean的定义上(例如提供给容易的配置元数据)也就是说Spring IoC容器允许BeanFactoryPostProcessor在容易实际实例化任何bean之前读取配置元数据并可能修改它

Spring包含了许多已有的bean工厂后置处理器例如PropertyResourceConfigure和PropertyPlaceHolderConfigure以及 BeanNameAutoProxyCreator

在一个BeanFactory中应用BeanFactoryPostProcessor需要手工编码实现而ApplicationContext则会检测部署在它之上实现了BeanFactoryPostProcessor接口的bean并在适当的时候自动把它们用做bean工作后置处理器部署一个后置处理器同部署其他的bean并没有什么区别无需其他的动作

PropertyPlaceholderConfigurer

作为一个bean工厂后置处理器的实现可以用来将BeanFactory定义中的一些属性值放置到另一个单独的Java Properties格式的文件中这就允许用户在部署应用的时候定制一些关键属性(例如数据库URL用户名和密码)而不用对主XML定义文件或容器所用文件进行复杂和危险的修改

PropertyOverrideConfigurer

类似于PropertyPlaceholderConfigurer但是与后者相比前者对于bean属性可以有却兴致或者根本没有值如果起覆盖左右的Properties文件没有某个bean属性的内容那么将使用却行的上下文定义

bean工厂的定义并不会议室到被覆盖所以仅仅擦看XML定义文件并不能立刻明显的知道覆盖配置是否被起作用了在多个PropertyOverrideConfigurer对一个bean属性定义了不同的值的时候最后一个将取胜

***他使用beanNamepropertyName来指定值而且不需要在bean中进行配置

注册用户自定义的PropertyEditors

a当用字符串值设置bean的属性时BeanFactory实质上使用了标准的JavaBeans的PropertyEditor将这些String转换为属性的复杂类型Spring预先注册了很多定制的PropertyEditor(比如将一个字符串表示我的classname转换成阵阵的Class对象)

b 要编写一个属性编辑器可以实现PropertyEditor接口更为简便的方式是从PropertyEditorSupport类继承

要使用自定义的PropertyEditors必须使用orgspringframewonfigCustomEditorConfigurer来注册自定义的属性编辑器

FactoryBean可以用来做代理

orgspringframeworkbeansfactoryFactoryBean

**要想得到FactoryBean本身需要在beanName前面加上&即&beanName

*********BeanFactory和ApplicationContext的区别*********

ApplicationContext是在ApplicationContext初始化的时候就把所有的bean都创建好了并存放在缓存中

BeanFactory是在需要得到bean的时候才去创建他的实例

上一篇:JUnit的各种断言

下一篇:hibernate事务,一级缓存,二级缓存