您的位置:首页 > 编程语言 > Java开发

spring源码初步学习-容器(BeanFactory)基本实现

2015-06-14 17:05 916 查看
第一部分:容器的基本实现

一 基本用法

配置文件 beanFactoryTest.xml

<span style="font-size:14px;"><?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation=" http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.1.xsd">

<bean id="myTestBean" class="com.zhangyiwen.website03.base_exercise.spring.MyTestBean"/>
</beans></span>


java代码

public void testSimpleLoad(){
BeanFactory bf = new XmlBeanFactory(new ClassPathResource(
            "beanFactoryTest.xml"));
    MyTestBean bean = (MyTestBean) bf.getBean("myTestBean");
    System.out.println(bean.getTestStr());
}


以上代码的基本功能

a.读取配置文件beanFactoryTest.xml

b.根据beanFactoryTest.xml中的位置找到对应的类的配置,并实例化

c.调用实例化后的实例

二 核心类介绍

2.1 DefaultListableBeanFactory

类结构图如下

    


1)DefaultListableBeanFactory继承AbstractAutowireCapableBeanFactory,并实现了ConfigurableListableBeanFactory和BeanDefinitionRegistry

2)AbstractAutowireCapableBeanFactory:综合AbstractBeanFactory并对接口AutowireCapableBeanFactory进行实现

AbstractBeanFactory:对单例的注册及获取,对FactoryBean的特殊处理,提供配置Factory的各种方法

AutowireCapableBeanFactory:提供创建Bean、自动注入、初始化以及应用bean的后处理器的接口

3)ConfigurableListableBeanFactory:BeanFactory配置清单等

4)BeanDefinitionRegistry定义对BeanDefinition的各种增删改操作

2.2 XmlBeanFactory

对DefaultListableBeanFactory进行了扩展,主要用于从XML文档中读取BeanDefinition

唯独与父类不同的是增加了XmlBeanDefinitionReader类型的reader属性,用于对资源文件进行读取、解析和注册

2.3 XmlBeanDefinitionReader

Spring的大部分功能都是以配置作为切入点的,该类就是用于对资源文件进行读取、解析和注册。

0)大致脉络

资源文件路径->Resource文件->Document文件->解析Document及Element

1)该类继承了AbstractBeanDefinitionReader,通过继承父类中的方法,使用ResourceLoader将资源文件路径转换为对应的Resource文件



2)通过自己包含的DocumentLoader对Resource资源进行转换,将Resource文件转换为Document文件

3)通过自己包含的DefaultBeanDefinitionDocumentReader类对Document进行解析

三 容器的基础XmlBeanFactory

3.1 配置文件封装

BeanFactory bf = new XmlBeanFactory(new ClassPathResource("beanFactoryTest.xml"));


在java中,将不同来源的资源抽象成URL,通过注册不同的handler(URLStreamHandler)来处理不同来源的资源的读取逻辑。

Spring对其内部使用到的资源实现了自己的抽象结构:Resource接口来封装底层资源。

Spring的配置文件读取是通过进行封装的。



InputStreamSource封装任何能返回InputStream的类

Resource接口抽象了所有Spring内部使用到的底层资源:File、URL、Classpath等

1)它定义了3个判断当前资源状态的方法:存在性(exists)、可读性(isReadable)、打开状态(isOpen)

2)Resource接口还提供了不同资源到URL、URI、File类型的转换

当通过Resource相关类完成了对配置
4000
文件的封装后,配置文件的读取工作就全权交给XmlBeanDefinitionReader来处理了。

3.2 加载Bean

通过调用XmlBeanDefinitionReader类型的reader的loadBeanDefinitions(Resource resource)方法,来开始整个资源的加载:

1)对传入的resource参数做封装,目的是考虑到Resource可能存在编码要求

new EncodedResource(resource)

2)通过SAX读取XML文件的方式准备InputSource对象

InputStream inputStream = encodedResource.getResource().getInputStream();

InputSource inputSource = new InputSource(inputStream);

3)将顺便的数据通过参数传入真正的核心处理部分doLoadBeanDefinitions

doLoadBeanDefinitions(inputSource, encodedResource.getResource());

核心部分
doLoadBeanDefinitions(inputSource, encodedResource.getResource()):

1)获取对XML文件的验证模式(DTD或XSD)

int validationMode = getValidationModeForResource(resource);

2)加载XML文件,得到对应的Document
Document doc = this.documentLoader.loadDocument(inputSource, getEntityResolver(), this.errorHandler, validationMode, isNamespaceAware());
3)根据返回的Document注册Bean信息
registerBeanDefinitions(doc, resource);
以上三个步骤支撑着整个Spring容器部分的实现基础。

解析及注册BeanDefinitions:registerBeanDefinitions(doc, resource);
1)实例化BeanDefinitionDocumentReader
BeanDefinitionDocumentReader documentReader = createBeanDefinitionDocumentReader();
BeanDefinitionDocumentReader 是一个接口,真正的实例类型是DefaultBeanDefinitionDocumentReader

2)documentReader.registerBeanDefinitions(doc, createReaderContext(resource))的功能
a.提取root:Element root = doc.getDocumentElement();

b.再次将root作为参数继续BeanDefinition的注册:doRegisterBeanDefinitions(root);-->注册Bean信息的真正核心

<span style="font-size:12px;">	protected void doRegisterBeanDefinitions(Element root) {
//1 处理profile属性
                String profileSpec = root.getAttribute(PROFILE_ATTRIBUTE);
if (StringUtils.hasText(profileSpec)) {
Assert.state(this.environment != null, "Environment must be set for evaluating profiles");
String[] specifiedProfiles = StringUtils.tokenizeToStringArray(
profileSpec, BeanDefinitionParserDelegate.MULTI_VALUE_ATTRIBUTE_DELIMITERS);
if (!this.environment.acceptsProfiles(specifiedProfiles)) {
return;
}
}

//专门处理解析
BeanDefinitionParserDelegate parent = this.delegate;
this.delegate = createHelper(this.readerContext, root, parent);

preProcessXml(root);
//2 解析并注册BeanDefinition
/*
对于根节点或者子节点,如果是默认命名空间的话则采用parseDefaultElement(ele, delegate)方法进行解析
否则使用delegate.parseCustomElement(ele)方法对自定义明明空间进行解析
*/
 <strong>parseBeanDefinitions(root, this.delegate);</strong>
 postProcessXml(root);

this.delegate = parent;
}</span>


对于默认标签解析与自定义标签解析,将在下一章博客继续...

四 默认标签的解析——以默认标签<bean>标签的解析和注册为例

4.1 解析BeanDefinition

Beandefinition是一个接口,是配置文件<bean>元素标签在容器中的内部表示形式。
其中RootBeanDefinition是最常用的实现类,它对应一般性的<bean>元素标签;GenericBeanDefinition是一站式服务类。两者都继承了AbstractBeanDefinition

Spring通过BeanDefinition将配置文件中的<bean>配置信息转换为容器的内部表示,并将这些BeanDefinition注册到BeanDefinitionRegistry中。
Spring容器的BeanDefinitionRegistry就像是Spring配置信息的内存数据库,主要是以map的形式保存,后续操作直接从BeanDefinitionRegistry中读取配置信息

4.2 注册解析的BeanDefinition
public static void registerBeanDefinition(BeanDefinitionHolder definitionHolder, BeanDefinitionRegistry registry)
1)通过BeanName的注册
a.对AbstractBeanDefinition进行校验
b.对beanName已经注册的情况进行处理(抛出异常或者直接覆盖)
c.加入map缓存:将beanDefinition直接放入map中,使用beanName作为key
d.清除解析之前留下的对应beanName的缓存

2)通过别名的注册

a.处理一些特殊情况:alias与beanName相同、alias覆盖处理、alias循环检查
b.注册alias

4.3 通知监听器解析及注册完成
getReaderContext().fireComponentRegistered(new BeanComponentDefinition(bdHolder));
开发人员可以通过注册监听器的方式将处理逻辑写入监听器中,则注册BeanDefinition事件发生时,处理逻辑就会执行

补充 spring相关使用示例
5.1
constructor-arg使用
配置文件
<bean id="helloBean2" class="com.zhangyiwen.website03.base_exercise.spring.constructor_arg.HelloBean">
<constructor-arg index="0"><value>张益文</value></constructor-arg>
<constructor-arg index="1"><value>你好</value></constructor-arg>
</bean>

示例Bean
public class HelloBean {

private String name;
private String word;

public HelloBean() {
}

public HelloBean(String name, String word) {
this.name = name;
this.word = word;
}

@Override
public String toString() {
return "HelloBean [name=" + name + ", word=" + word + "]";
}

}

测试代码
HelloBean bean = (HelloBean) bf.getBean("helloBean2");
System.out.println(bean.toString());

输出结果
HelloBean [name=张益文, word=你好]


5.2 import标签的使用
配置文件
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation=" http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.1.xsd">
<import resource="customerContext.xml" />
<import resource="systemContext.xml" />
</beans>applicationContext.xml文件中使用import的方式导入有关模块配置文件,以后若有新模块加入,就可以简单修改这个文件了
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  spring