`
liu75675231
  • 浏览: 3341 次
文章分类
社区版块
存档分类
最新评论

菜鸟的spring 3.0源码学习之旅(5)

阅读更多
今天天气大范围降温,或许对于南方还是阳春三月的气候,但是对于我们这里,就将要进入千里冰封万里雪飘的季节了,不仅如此,我柔弱的手机也禁不住这大范围的降温也因此进入了休眠状态
不过阵阵的寒意毅然挡不住我学习的热情,我下定决心,在我变身成为冰雕之前,一定要先把spring的源码搞懂,因为吧spring都搞懂了,春天还远吗
不瞎扯了,言归正传吧
昨天呢,我们讲述了XmlBeanFactory的字段实例化所发生的动作,那么今天呢,我们主要是讨论一下构造器里发生的一切,
首先,我们再看一下这个构造器
public XmlBeanFactory(Resource resource, BeanFactory parentBeanFactory) throws BeansException {
super(parentBeanFactory);
this.reader.loadBeanDefinitions(resource);  //加载BeanDefinitions信息
}
在这个构造器里,第一句的意思是说,吧parentBeanFactory交给父类的构造器来处理,
public AbstractAutowireCapableBeanFactory() {
super();
ignoreDependencyInterface(BeanNameAware.class);
ignoreDependencyInterface(BeanFactoryAware.class);
ignoreDependencyInterface(BeanClassLoaderAware.class);
}
public AbstractAutowireCapableBeanFactory(BeanFactory parentBeanFactory) {
this();
setParentBeanFactory(parentBeanFactory);
}
在这个构造器里,我们可以看到,构造器里实际上做了两个工作,一个是调用ignoreDependencyInterface,一个是调用了其父类的setParentBeanFactory,第一个工作实际上是往private final Set<Class> ignoredDependencyInterfaces = new HashSet<Class>();这个字段里设置添加参数,第二个呢,是吧这个入参设置到父类的私有字段里
不过呢,当这个类在实例化的同时,这个静态代码会先获得执行
private static Class javaxInjectProviderClass = null;
static {
ClassLoader cl = DefaultListableBeanFactory.class.getClassLoader();
try {
javaxInjectProviderClass = cl.loadClass("javax.inject.Provider");
}
catch (ClassNotFoundException ex) {
}
}
这个是为该字段动态的添加类实例信息,这个我到目前还没有发现有什么有用的地方,所以咱们暂时先忽略这个地方
XmlBeanFactory构造器里第一行代码就执行完了,那么我们就开始研究第二行代码吧
this.reader.loadBeanDefinitions(resource);
这个呢,是XmlBeanFactory的字段调用了loadBeanDefinition方法,那么,我们就欣赏一下这个方法的实现吧
org.springframework.beans.factory.xml.XmlBeanDefinitionReader#loadBeanDefinitions
public int loadBeanDefinitions(Resource resource) throws BeanDefinitionStoreException {
return loadBeanDefinitions(new EncodedResource(resource));
}
这个方法的主要作用是吧Resource封装成EncodedResource类,这个EncodedResource类呢,有两个字段,一个是字符串类型的编码格式字段,另一个是Resource信息,这样大家应该就了解的差不多了吧,所以也不再赘述,我们然后再往下看
org.springframework.beans.factory.xml.XmlBeanDefinitionReader#loadBeanDefinitions
public int loadBeanDefinitions(EncodedResource encodedResource) throws BeanDefinitionStoreException {
Assert.notNull(encodedResource, "EncodedResource must not be null");
if (logger.isInfoEnabled()) {
logger.info("Loading XML bean definitions from " + encodedResource.getResource());
}

Set<EncodedResource> currentResources = this.resourcesCurrentlyBeingLoaded.get();
if (currentResources == null) {  //如果变量为null,则实例化,并添加到本地线程中
currentResources = new HashSet<EncodedResource>(4);
this.resourcesCurrentlyBeingLoaded.set(currentResources);
}
if (!currentResources.add(encodedResource)) {   //如果currentResources包含encodedResource,则抛出异常
throw new BeanDefinitionStoreException(
"Detected cyclic loading of " + encodedResource + " - check your import definitions!");
}
try {
InputStream inputStream = encodedResource.getResource().getInputStream();
try {
InputSource inputSource = new InputSource(inputStream);
if (encodedResource.getEncoding() != null) {
inputSource.setEncoding(encodedResource.getEncoding());
}
return doLoadBeanDefinitions(inputSource, encodedResource.getResource());
}
finally {
inputStream.close();
}
}
catch (IOException ex) {
throw new BeanDefinitionStoreException(
"IOException parsing XML document from " + encodedResource.getResource(), ex);
}
finally {
currentResources.remove(encodedResource);
if (currentResources.isEmpty()) {
this.resourcesCurrentlyBeingLoaded.remove();
}
}
}
这个方法貌似比较复杂,不过我们别着急,看我耐着心给大家分析
Set<EncodedResource> currentResources = this.resourcesCurrentlyBeingLoaded.get();
if (currentResources == null) {  //如果变量为null,则实例化,并添加到本地线程中
currentResources = new HashSet<EncodedResource>(4);
this.resourcesCurrentlyBeingLoaded.set(currentResources);
}
if (!currentResources.add(encodedResource)) {   //如果currentResources包含encodedResource,则抛出异常
throw new BeanDefinitionStoreException(
"Detected cyclic loading of " + encodedResource + " - check your import definitions!");
}
这个主要是往字段中添加资源,这个字段是用本地线程包装的,我们来看这个字段
private final ThreadLocal<Set<EncodedResource>> resourcesCurrentlyBeingLoaded =
new NamedThreadLocal<Set<EncodedResource>>("XML bean definition resources currently being loaded");
然后我让大家看下面的这个语句段
InputStream inputStream = encodedResource.getResource().getInputStream();
try {
InputSource inputSource = new InputSource(inputStream);
if (encodedResource.getEncoding() != null) {
inputSource.setEncoding(encodedResource.getEncoding());
}
return doLoadBeanDefinitions(inputSource, encodedResource.getResource());
}
finally {
inputStream.close();
}
InputSource是对要解析的XML文件的一个抽象,资源返回输出流,然后由InputSource实例化,如果资源的编码方式不为null,则设置其格式,然后将会执行doLoadBeanDefinitions方法
protected int doLoadBeanDefinitions(InputSource inputSource, Resource resource)
throws BeanDefinitionStoreException {
try {
int validationMode = getValidationModeForResource(resource);  //XSD
Document doc = this.documentLoader.loadDocument(
inputSource, getEntityResolver(), this.errorHandler, validationMode, isNamespaceAware()); 
return registerBeanDefinitions(doc, resource);
}
catch (BeanDefinitionStoreException ex) {
throw ex;
}
catch (SAXParseException ex) {
throw new XmlBeanDefinitionStoreException(resource.getDescription(),
"Line " + ex.getLineNumber() + " in XML document from " + resource + " is invalid", ex);
}
catch (SAXException ex) {
throw new XmlBeanDefinitionStoreException(resource.getDescription(),
"XML document from " + resource + " is invalid", ex);
}
catch (ParserConfigurationException ex) {
throw new BeanDefinitionStoreException(resource.getDescription(),
"Parser configuration exception parsing XML from " + resource, ex);
}
catch (IOException ex) {
throw new BeanDefinitionStoreException(resource.getDescription(),
"IOException parsing XML document from " + resource, ex);
}
catch (Throwable ex) {
throw new BeanDefinitionStoreException(resource.getDescription(),
"Unexpected exception parsing XML document from " + resource, ex);
}
}
这个是本类的doLoadBeanDefinitions方法,在这个方法里,主要是吧InputSource还有资源文件解析成Document,我想这个大家应该一目了然,也不需要多说,
int validationMode = getValidationModeForResource(resource);
在此方法中的这个字段是为了获得validationMode的值,在这里,此值默认为xsd,
getEntityResolver()此方法是为了实例化EntityResolver,
不过,在这个方法里,真正的重点是这个
Document doc = this.documentLoader.loadDocument(
inputSource, getEntityResolver(), this.errorHandler, validationMode, isNamespaceAware()); 
这个字段正是此方法的核心所在,那么我们看看loadDocument方法都进行了什么处理了
public Document loadDocument(InputSource inputSource, EntityResolver entityResolver,
ErrorHandler errorHandler, int validationMode, boolean namespaceAware) throws Exception {

DocumentBuilderFactory factory = createDocumentBuilderFactory(validationMode, namespaceAware);//实例化此类,并且把验证方式设为dtd验证,支持名称空间
if (logger.isDebugEnabled()) {
logger.debug("Using JAXP provider [" + factory.getClass().getName() + "]");
}
DocumentBuilder builder = createDocumentBuilder(factory, entityResolver, errorHandler);   //实例化该类,并且指定使用 EntityResolver 解析要解析的 XML 文档中存在的实体,使用指定解析器要使用的 ErrorHandler
return builder.parse(inputSource);
}
大家看了这个方法,我想都明白了吧,我也就不多说什么了,接着继续
当执行完doLoadBeanDefinitions以后,紧接着就要执行registerBeanDefinitions方法了
public int registerBeanDefinitions(Document doc, Resource resource) throws BeanDefinitionStoreException {
BeanDefinitionDocumentReader documentReader = createBeanDefinitionDocumentReader();  //实例化BeanDefinitionDocumentReader
int countBefore = getRegistry().getBeanDefinitionCount();
documentReader.registerBeanDefinitions(doc, createReaderContext(resource));
return getRegistry().getBeanDefinitionCount() - countBefore;
}
在这个方法里,将完成对Document的解析的工作,而这个方法的核心是这个字段
documentReader.registerBeanDefinitions(doc, createReaderContext(resource));
那么我们就主要看一下registerBeanDefinition方法吧
org.springframework.beans.factory.xml. DefaultBeanDefinitionDocumentReader#registerBeanDefinitions
public void registerBeanDefinitions(Document doc, XmlReaderContext readerContext) {
this.readerContext = readerContext;

logger.debug("Loading bean definitions");
Element root = doc.getDocumentElement();   //获得根元素

BeanDefinitionParserDelegate delegate = createHelper(readerContext, root);   //初始化bean中的信息

preProcessXml(root);   //空方法
parseBeanDefinitions(root, delegate);
postProcessXml(root);
}
那么我们看看createHelper方法都定义了什么吧
protected BeanDefinitionParserDelegate createHelper(XmlReaderContext readerContext, Element root) {
BeanDefinitionParserDelegate delegate = new BeanDefinitionParserDelegate(readerContext);
delegate.initDefaults(root);  //初始化bean中的属性
return delegate;
}
在这个方法里,它的核心是这个字段
delegate.initDefaults(root);
那么我们再看看initDefaults方法吧
public void initDefaults(Element root) {
populateDefaults(this.defaults, root);
this.readerContext.fireDefaultsRegistered(this.defaults);   //尚不清楚作用
}
下面的这个方法主要对资源中的各个属性取出,设置到DocumentDefaultsDefinition中,
protected void populateDefaults(DocumentDefaultsDefinition defaults, Element root) {
defaults.setLazyInit(root.getAttribute(DEFAULT_LAZY_INIT_ATTRIBUTE));
defaults.setMerge(root.getAttribute(DEFAULT_MERGE_ATTRIBUTE));
defaults.setAutowire(root.getAttribute(DEFAULT_AUTOWIRE_ATTRIBUTE));
defaults.setDependencyCheck(root.getAttribute(DEFAULT_DEPENDENCY_CHECK_ATTRIBUTE));
if (root.hasAttribute(DEFAULT_AUTOWIRE_CANDIDATES_ATTRIBUTE)) {
defaults.setAutowireCandidates(root.getAttribute(DEFAULT_AUTOWIRE_CANDIDATES_ATTRIBUTE));
}
if (root.hasAttribute(DEFAULT_INIT_METHOD_ATTRIBUTE)) {
defaults.setInitMethod(root.getAttribute(DEFAULT_INIT_METHOD_ATTRIBUTE));
}
if (root.hasAttribute(DEFAULT_DESTROY_METHOD_ATTRIBUTE)) {
defaults.setDestroyMethod(root.getAttribute(DEFAULT_DESTROY_METHOD_ATTRIBUTE));
}
defaults.setSource(this.readerContext.extractSource(root));    //尚不清楚作用
}
好了,讲到这里,我就把XmlBeanFactory的实例化过程大体上给梳理了一遍,本人愚笨,用了两天的时间才把其中的缘由搞明白,我想大家都比我聪明,所以还应多请教为是

分享到:
评论

相关推荐

    python3.0菜鸟教程.zip

    作为初学者,了解并掌握这些基础知识对于学习 Python 3.0 至关重要。以下是一些关键的知识点,它们在 "www.runoob.com" 提供的菜鸟教程中应该有所涵盖: 1. **基础语法**: - **变量与数据类型**:Python 支持整型...

    Spring3.0的新特性

    马士兵老师提供的讲解,对于菜鸟或者是刚学习javaweb和SSH框架的人来说可谓是福音,马老师讲的内容非常详细,同时还会灌输学者一些在实际工作和开发中的一些技巧,特别是代码规范和编程习惯是非常重要的,所以特意...

    python菜鸟3.0-终于清楚python菜鸟入门教程.pdf

    总的来说,Python的学习是一个循序渐进的过程,从环境配置、基础语法学习,到实践操作和解决问题,每一步都是为了更好地掌握这门语言。保持耐心,多动手实践,你将成为一名熟练的Python程序员。

    菜鸟 Spring 源码解读 推荐流程

    在本文中,我们将深入探讨Spring的源码,尤其是针对"菜鸟"级别的开发者如何逐步理解并掌握Spring的核心机制。 首先,源码解读对于任何开发者来说都是提升技术能力的重要途径。通过阅读源码,我们可以更直观地了解...

    spring5源码学习

    在这个系列的学习中,我们将深入探讨Spring5的核心原理,通过分析源码来揭示其内在的工作机制,并结合实战应用,提升对Spring5的理解和运用能力。 一、Spring5新特性 1. 支持Java 9及以上的版本:Spring5全面支持...

    struts2+hibernate3.2+spring3.0实现简单的增删改查

    struts2+hibernate3.2+spring3.0实现简单的增删改查

    34--[菜鸟闯关].zip源码scratch2.0 3.0编程项目源文件源码案例素材源代码

    34--[菜鸟闯关].zip源码scratch2.0 3.0编程项目源文件源码案例素材源代码34--[菜鸟闯关].zip源码scratch2.0 3.0编程项目源文件源码案例素材源代码34--[菜鸟闯关].zip源码scratch2.0 3.0编程项目源文件源码案例素材源...

    springcloud简单项目搭建,菜鸟学习

    springcloud简单项目搭建,菜鸟学习Spring Cloud集成相关优质项目推荐 这些项目是Spring Cloud官方项目或是对Spring Cloud进行了有益的补充以及基于Spring Cloud最佳实践。

    spring菜鸟入门经典实例

    本教程专为初学者设计,旨在帮助“菜鸟”快速掌握Spring的基本概念和实战技巧。 在Spring入门阶段,首先要理解的是依赖注入(DI)。DI是一种设计模式,它允许我们解耦组件,让它们之间通过接口而非具体的实现进行...

    菜鸟窝企业项目课程《菜鸟商城》的源码.rar

    通过学习和实践《菜鸟商城》项目,开发者不仅可以掌握Java后端和安卓客户端的开发技术,还能深入了解电商系统的架构设计和业务逻辑,为今后的项目开发积累宝贵经验。同时,该项目也适合初学者进行实战演练,提升自己...

    spring整合redis

    在IT行业中,Spring框架是Java应用开发的基石,它提供了丰富的功能来简化企业级应用的构建。Mybatis作为一款轻量级的持久层框架,与Spring的集成使得数据访问更加高效便捷。而Redis则是一款高性能的内存数据结构存储...

    深入Python学习:基于菜鸟教程的实战源码解析

    项目概述:深入Python学习实战源码解析 本项目以Python为主要开发语言,基于菜鸟教程的指导,深入探索和实践Python编程。项目包含62个文件,其中38个为...简洁描述:基于菜鸟教程的Python学习实践记录与源码解析集锦。

    仿XP网站源码(PHP200菜鸟)dzx3.0版

    5、templates_c smarty自动生成的文件 6、JS 常用的JS(当时顺手放的) 7、other 乱78遭的东西,以前学HTML时做的 8、html 生成的HTML文件自动放此处 主要文件 说明: controller 文件夹中 usercontroller.php ...

    易语言源码易语言菜鸟背包游戏源码.rar

    易语言源码易语言菜鸟背包游戏源码.rar 易语言源码易语言菜鸟背包游戏源码.rar 易语言源码易语言菜鸟背包游戏源码.rar 易语言源码易语言菜鸟背包游戏源码.rar 易语言源码易语言菜鸟背包游戏源码.rar 易语言源码...

    Linux扫盲-Linux菜鸟过关(V3.0)——Linux入门必备

    您不需要任何Linux技能也能阅读本教程,Linux菜鸟过关全面解决您的Linux之旅问题。您现在要做的是了解一些Linux知识与安装方法,按照本教程的顺序阅读即可。本教程可以作为您短期突破Linux安装的教材,这一切都不...

    仿菜鸟商城系统(源码)

    【仿菜鸟商城系统(源码)】是一款专为学习目的设计的电商系统,它提供了从用户界面到后台管理的一整套功能,可以帮助开发者理解和实践电子商务平台的构建过程。该系统模仿了真实的在线购物平台——菜鸟网络,因此在...

    51菜鸟到arm高手

    51菜鸟到arm高手51菜鸟到arm高手51菜鸟到arm高手51菜鸟到arm高手51菜鸟到arm高手51菜鸟到arm高手51菜鸟到arm高手51菜鸟到arm高手

    Spring的源码解析

    对于spring源码的深度解析,适合菜鸟向中级进阶的时候看看

    android商城源码(菜鸟商城)

    总的来说,“菜鸟商城”源码是一个功能完备、注释详尽的Android商城应用实例,对于学习Android开发和实际项目经验的积累都有很大帮助。通过深入研究这份源码,开发者不仅可以掌握Android应用开发的基本技能,还能...

    (3.0版本)自己写的struts2+hibernate+spring实例

    jar包:项目中除了使用了struts2,hibernate3.0和spring1.8以外(用spring2.0的包也可以.不能低于1.8的包.)还是用了junit,ajax,第三方的table组件.等等.所以需要下载相对应的包. 为了上传jar.我专门申请了一个网盘.所有...

Global site tag (gtag.js) - Google Analytics