Spring 5.2.9 06 源码分析-spring自定义标签解析过程(上)

完整路径:

//程序入口
MyClassPathXmlApplicationContext.MyClassPathXmlApplicationContext(String... configLocations){}
ClassPathXmlApplicationContext(String... configLocations) throws BeansException {}
ClassPathXmlApplicationContext.ClassPathXmlApplicationContext(String... configLocations) throws BeansException {}
this(configLocations, true, null);
		#重复方法, enter
		this(configLocations, true, null);
			refresh();
#spring核心代码#refresh();
//Spring源码核心,所有初始化信息的加载过程在全部都在这里。
AbstractApplicationContext.refresh() throws BeansException, IllegalStateException {}
			// 加载xml配置文件的属性值到当前工厂中,最重要的就是BeanDefinition
			ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();
	#重复方法, enter
		// 初始化BeanFactory,并进行XML文件读取,并将得到的BeanFactory记录在当前实体的属性中
		refreshBeanFactory();
	#注意选择子类实现
	protected abstract void refreshBeanFactory() throws BeansException, IllegalStateException;
AbstractRefreshableApplicationContext.refreshBeanFactory() throws BeansException, IllegalStateException;
			// 初始化documentReader,并进行XML文件读取及解析,默认命名空间的解析,自定义标签的解析
			loadBeanDefinitions(beanFactory);
	//注意选择子类实现
	protected abstract void loadBeanDefinitions(DefaultListableBeanFactory beanFactory) throws BeansException,IOException;
AbstractXmlApplicationContext.loadBeanDefinitions(DefaultListableBeanFactory beanFactory) throws BeansException, IOException {}
		// 开始完成beanDefinition的加载
		loadBeanDefinitions(beanDefinitionReader);
			#方法调用,进入
			reader.loadBeanDefinitions(configResources);
AbstractBeanDefinitionReader.loadBeanDefinitions(Resource... resources) throws BeanDefinitionStoreException {}
				count += loadBeanDefinitions(resource);
BeanDefinitionReader.loadBeanDefinitions(Resource resource) throws BeanDefinitionStoreException;
XmlBeanDefinitionReader.loadBeanDefinitions(Resource resource) throws BeanDefinitionStoreException {}
				#重复代码
				return loadBeanDefinitions(location, null);
AbstractBeanDefinitionReader
				#方法
				loadBeanDefinitions(String location, @Nullable Set<Resource> actualResources) throws BeanDefinitionStoreException {}
				#代码
				int count = loadBeanDefinitions(resources);
				#代码
				return loadBeanDefinitions(location, null);
				#方法
				loadBeanDefinitions(String location, @Nullable Set<Resource> actualResources) throws BeanDefinitionStoreException {}
				#代码
				int count = loadBeanDefinitions(resources);
				#方法
				loadBeanDefinitions(Resource... resources) throws BeanDefinitionStoreException {
				count += loadBeanDefinitions(resource);
BeanDefinitionReader.loadBeanDefinitions(Resource resource) throws BeanDefinitionStoreException;
XmlBeanDefinitionReader.loadBeanDefinitions(Resource resource) throws BeanDefinitionStoreException {}
XmlBeanDefinitionReader.loadBeanDefinitions(EncodedResource encodedResource) throws BeanDefinitionStoreException {}
			// 逻辑处理的核心步骤
			return doLoadBeanDefinitions(inputSource, encodedResource.getResource());
			doLoadBeanDefinitions(InputSource inputSource, Resource resource) throws BeanDefinitionStoreException {}
			#方法
			int count = registerBeanDefinitions(doc, resource);
			registerBeanDefinitions(Document doc, Resource resource) throws BeanDefinitionStoreException {
			documentReader.registerBeanDefinitions(doc, createReaderContext(resource));
XmlBeanDefinitionReader.registerBeanDefinitions(Document doc, XmlReaderContext readerContext) {}
		doRegisterBeanDefinitions(doc.getDocumentElement());
		protected void doRegisterBeanDefinitions(Element root) {}
		#多次调用代码
		parseBeanDefinitions(root, this.delegate);
		protected void parseBeanDefinitions(Element root, BeanDefinitionParserDelegate delegate) {}
						//解析额外标签
						delegate.parseCustomElement(ele);
XmlBeanDefinitionReade.parseCustomElement(Element ele) {}
	return parseCustomElement(ele, null);
XmlBeanDefinitionReader.parseCustomElement(Element ele, @Nullable BeanDefinition containingBd) {}
		// 调用自定义的NamespaceHandler进行解析
		return handler.parse(ele, new ParserContext(this.readerContext, this, containingBd));
NamespaceHandler.parse(Element element, ParserContext parserContext);
		return (parser != null ? parser.parse(element, parserContext) : null);
NamespaceHandlerSupport.parse(Element element, ParserContext parserContext) {}
BeanDefinitionParser.parse(Element element, ParserContext parserContext);
#注意选择子类实现
#路径结束#
ComponentScanBeanDefinitionParser.parse(Element element, ParserContext parserContext) {}

解析额外标签的核心代码。

	@Override
	@Nullable
	public BeanDefinition parse(Element element, ParserContext parserContext) {
		// 获取<context:component-scan>节点的base-package属性值
		String basePackage = element.getAttribute(BASE_PACKAGE_ATTRIBUTE);
		// 解析占位符
		basePackage = parserContext.getReaderContext().getEnvironment().resolvePlaceholders(basePackage);
		// 解析base-package(允许通过,;\t\n中的任一符号填写多个)
		String[] basePackages = StringUtils.tokenizeToStringArray(basePackage,
				ConfigurableApplicationContext.CONFIG_LOCATION_DELIMITERS);

		// Actually scan for bean definitions and register them.
		// 构建和配置ClassPathBeanDefinitionScanner
		ClassPathBeanDefinitionScanner scanner = configureScanner(parserContext, element);
		// 使用scanner在执行的basePackages包中执行扫描,返回已注册的bean定义
		Set<BeanDefinitionHolder> beanDefinitions = scanner.doScan(basePackages);
		// 组件注册(包括注册一些内部的注解后置处理器,触发注册事件)
		registerComponents(parserContext.getReaderContext(), beanDefinitions, element);

		return null;
	}

注释说明

What
Why
How
我本就只想记录一下,调用放大的记录。看Spring源码会点着点着不知道点到那里了。就好像分支树一样,我需要一个可回归的记录树。
1.第一个阶段是记录。
2.第二阶段是分析
3.第三个阶段是搜索。

我现在处于第一个阶段,我需要先记录。
等到存储到了足够的范围。我想要找到树结构的软件,支持检索。方便我以后回顾浏览。
range:Spring配置文件解析(yml,properties,xml),AOP,IOC,BeanFactory

方案一:
使用xmind7,思维导图的方式查询,但是我发现我不喜欢点点,搜索的精确性不支持标签匹配。
最开始我还尝试使用截图保存,梳理方法内部代码。导致了内容不支持搜索。

!!!我现在遇到的障碍。
路径追求的是单一路线完整分支,通过采用example示例的方式。这是第一个版本需要做的事情。
第二阶段则需要构造完整的结构树,我怎么转换数据?
…我之前记录过一个版本,如果csdn记录路径需要2小时完成,思维导图记录需要1小时。查错回归需要2小时。
合计需要5小时…在经历过滤到第三版本,那么我需要的耗费的时间将会翻倍。
…我将会没有时间优化csdn内容排版。我思维上除了检索的功能,我还需要可以扩展标签搜索,分类导航。我的终点,我需要耗费多少时间?控制,耗时,概念…如何确定自己任务范围?

完整路径,有三类
一类:类名->最后调用的方法->最后调用的代码
二类:重复调用的方法->重复调用的代码
三类:抽象类,接口类子类实现选择

一类:
::不需要考虑注释,除了关键代码。默认第一个调用方法使用{}结束,左侧无缩进。‘
::第二类方法则使用{},采用链表的思维方式进行关联。
#spring核心代码#refresh();
#路径结束#
二类:
#重复方法
#重复代码
#方法调用
三类:
#注意选择子类实现

注释事项

实现类进入,不要通过调用方法进入。我最开始调试源码时候也是这样,然后一直找不到正确的进入方法。
在这里插入图片描述

实现类进入,正确的方法是通过子类实现进入。
IDEA开发工具下,点击文本编辑器左侧航航的大写I图标进入。
在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值