热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

ConfigurationClassPostProcessor如何放入processors中

1.什么时候使用到当使用到context:annotation-config或者context:component-scan配置的时候,及声明做为其它BeanFac




1. 什么时候使用到

当使用到 context:annotation-config/ 或者context:component-scan/配置的时候,及声明做为其它BeanFactoryPostProcessor的时候。
本例使用:

<context:component-scan base-package&#61;"com.songbl.config" >context:component-scan>

2.1 源码分析

容器启动&#xff0c;在解析配置文件的时候&#xff1a;

DefaultBeanDefinitionDocumentReader.java
->

protected void parseBeanDefinitions(Element root, BeanDefinitionParserDelegate delegate) {
if (delegate.isDefaultNamespace(root)) {
NodeList nl &#61; root.getChildNodes();
for (int i &#61; 0; i < nl.getLength(); i&#43;&#43;) {
Node node &#61; nl.item(i);
if (node instanceof Element) {
Element ele &#61; (Element) node;
if (delegate.isDefaultNamespace(ele)) {
parseDefaultElement(ele, delegate);
}
else {
//调用其它自定义解析器。
//根据命名空间获取处理器&#xff0c;在处理器中根据属性标签&#xff0c;获取解析器&#xff0c;解析
delegate.parseCustomElement(ele);
}
}
}
}
else {
delegate.parseCustomElement(root);
}
}
//&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;
//&#61;>
public BeanDefinition parseCustomElement(Element ele, &#64;Nullable BeanDefinition containingBd) {
// 获取对应的命名空间
String namespaceUri &#61; getNamespaceURI(ele);
if (namespaceUri &#61;&#61; null) {
return null;
}
// 根据命名空间找到对应的NamespaceHandler 接口->对象&#xff0c;
// 根据命名空间uri&#xff0c;找到value &#xff0c;再实例化处理器。处理器包含解析器&#xff0c;根据ele找到解析器
//内部有实例化的调用&#xff0c;实例化NamespaceHander
NamespaceHandler handler &#61; this.readerContext.getNamespaceHandlerResolver().resolve(namespaceUri);
if (handler &#61;&#61; null) {
error("Unable to locate Spring NamespaceHandler for XML schema namespace [" &#43; namespaceUri &#43; "]", ele);
return null;
}//举例使用ContextNamespaceHandler ,map里面有很多解析器
// 调用自定义的NamespaceHandler进行解析。通过findParserForElement&#xff0c;找到对应的解析器
//parse 是根据属性标签找到的
return handler.parse(ele, new ParserContext(this.readerContext, this, containingBd));
}
//&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;
// hander 实例化时候会注册一些解析器
//注册各种解析器&#xff0c;其中有ComponentScanBeanDefinitionParser
ContextNamespaceHandler.java
&#64;Override
public void init() {
registerBeanDefinitionParser("property-placeholder", new PropertyPlaceholderBeanDefinitionParser());
registerBeanDefinitionParser("property-override", new PropertyOverrideBeanDefinitionParser());
registerBeanDefinitionParser("annotation-config", new AnnotationConfigBeanDefinitionParser());
//&#61;&#61;&#61;&#61;&#61;用到扫描parse,就回加上InterXXconfigXXX&#61;&#61;&#61;&#61;&#61;
registerBeanDefinitionParser("component-scan", new ComponentScanBeanDefinitionParser());
registerBeanDefinitionParser("load-time-weaver", new LoadTimeWeaverBeanDefinitionParser());
registerBeanDefinitionParser("spring-configured", new SpringConfiguredBeanDefinitionParser());
....
}

spring.handlers&#xff1a;

http\://www.springframework.org/schema/context&#61;org.springframework.context.config.ContextNamespaceHandler

根据指定的xml&#xff0c;实例化对应的Hander&#xff0c;在实例化Hander的同时&#xff0c;也注册了解析器。假如application.xml文件是&#xff1a;

&#xff0c;那么上面的parse方法即开始调用对应类的parse(…)&#xff0c;本例使用ComponentScanBeanDefinitionParser。


开始调用&#xff1a; parse() 解析方法。

接上面的parse方法

ComponentScanBeanDefinitionParser.java

public BeanDefinition parse(Element element, ParserContext parserContext) {
// 获取节点的base-package属性值
String basePackage &#61; element.getAttribute(BASE_PACKAGE_ATTRIBUTE);
// 解析占位符
basePackage &#61; parserContext.getReaderContext().getEnvironment().resolvePlaceholders(basePackage);
// 解析base-package
String[] basePackages &#61; StringUtils.tokenizeToStringArray(basePackage,
ConfigurableApplicationContext.CONFIG_LOCATION_DELIMITERS);
// 构建和配置ClassPathBeanDefinitionScanner
ClassPathBeanDefinitionScanner scanner &#61; configureScanner(parserContext, element);
// 使用scanner在执行的basePackages包中执行扫描&#xff0c;返回已注册的bean定义
Set<BeanDefinitionHolder> beanDefinitions &#61; scanner.doScan(basePackages);
// 组件注册&#xff08;包括注册一些内部的注解后置处理器&#xff0c;触发注册事件&#xff09;。
//扫描&#xff0c;下面有internalXXConfigurationXX
registerComponents(parserContext.getReaderContext(), beanDefinitions, element);
return null;
}

2.1.1.1 doScan()-扫描

使用scanner扫描指定包下的类&#xff0c;拿到bean信息后进行注册。返回已注册的bean信息用于注册组件。

protected Set<BeanDefinitionHolder> doScan(String... basePackages) {
Assert.notEmpty(basePackages, "At least one base package must be specified");
Set<BeanDefinitionHolder> beanDefinitions &#61; new LinkedHashSet<>();
// 遍历basePackages
for (String basePackage : basePackages) {
// 扫描basePackage,将符合要求的bean定义全部找出来
Set<BeanDefinition> candidates &#61; findCandidateComponents(basePackage);
// 遍历所有候选的bean定义
for (BeanDefinition candidate : candidates) {
// 解析&#64;Scope注解&#xff0c;包括scopeName和proxyMode
ScopeMetadata scopeMetadata &#61; this.scopeMetadataResolver.resolveScopeMetadata(candidate);
candidate.setScope(scopeMetadata.getScopeName());
// 使用beanName生成器来生成beanName
String beanName &#61; this.beanNameGenerator.generateBeanName(candidate, this.registry);
if (candidate instanceof AbstractBeanDefinition) {
// 处理beanDefinition对象&#xff0c;例如&#xff0c;此bean是否可以自动装配到其他bean中
postProcessBeanDefinition((AbstractBeanDefinition) candidate, beanName);
}
if (candidate instanceof AnnotatedBeanDefinition) {
// 处理定义在目标类上的通用注解&#xff0c;包括&#64;Lazy&#xff0c;&#64;Primary&#xff0c;&#64;DependsOn&#xff0c;&#64;Role&#xff0c;&#64;Description
AnnotationConfigUtils.processCommonDefinitionAnnotations((AnnotatedBeanDefinition) candidate);
}
// 检查beanName是否已经注册过&#xff0c;如果注册过&#xff0c;检查是否兼容
if (checkCandidate(beanName, candidate)) {
// 将当前遍历bean的bean定义和beanName封装成BeanDefinitionHolder
BeanDefinitionHolder definitionHolder &#61; new BeanDefinitionHolder(candidate, beanName);
// 根据proxyMode的值&#xff0c;选择是否创建作用域代理
definitionHolder &#61;
AnnotationConfigUtils.applyScopedProxyMode(scopeMetadata, definitionHolder, this.registry);
beanDefinitions.add(definitionHolder);
// &#61;&#61;&#61;&#61;&#61;&#61;&#61;注册beanDefinition&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;
registerBeanDefinition(definitionHolder, this.registry);
}
}
}
return beanDefinitions;
}

2.1.1.2 注册组件

其中有一个重要的configurationclasspostprocessor的beanName

protected void registerComponents(
XmlReaderContext readerContext, Set<BeanDefinitionHolder> beanDefinitions, Element element) {
Object source &#61; readerContext.extractSource(element);
// 使用注解的tagName和source构建CompositeComponentDefinition
CompositeComponentDefinition compositeDef &#61; new CompositeComponentDefinition(element.getTagName(), source);
// 将扫描到的所有beanDefinition添加到compositeDef的nestedComponents属性中
for (BeanDefinitionHolder beanDefHolder : beanDefinitions) {
compositeDef.addNestedComponent(new BeanComponentDefinition(beanDefHolder));
}
// Register annotation config processors, if necessary.
boolean annotationConfig &#61; true;
if (element.hasAttribute(ANNOTATION_CONFIG_ATTRIBUTE)) {
// 获取component-scan标签的annotation-config属性值&#xff0c;默认为true
annotationConfig &#61; Boolean.parseBoolean(element.getAttribute(ANNOTATION_CONFIG_ATTRIBUTE));
}
if (annotationConfig) {
// 如果annotation-config属性值为true&#xff0c;在给定的注册表中注册所有用于注解的bean后置处理器
Set<BeanDefinitionHolder> processorDefinitions &#61;
AnnotationConfigUtils.registerAnnotationConfigProcessors(readerContext.getRegistry(), source);
for (BeanDefinitionHolder processorDefinition : processorDefinitions) {
// *******其中有一个重要的configurationclasspostprocessor******
// 将注册的注解后置处理器的BeanDefinition添加到compositeDef的nestedComponents属性中
compositeDef.addNestedComponent(new BeanComponentDefinition(processorDefinition));
}
}
// 触发组件注册事件&#xff0c;默认实现为EmptyReaderEventListener
readerContext.fireComponentRegistered(compositeDef);
}
//&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;
registerAnnotationConfigProcessors(.....){
.......
// 注册内部管理的用于处理&#64;configuration注解的后置处理器的bean
if (!registry.containsBeanDefinition(CONFIGURATION_ANNOTATION_PROCESSOR_BEAN_NAME)) {
RootBeanDefinition def &#61; new RootBeanDefinition(ConfigurationClassPostProcessor.class);
def.setSource(source);
// 注册BeanDefinition到注册表中&#xff0c;registerBeanPostProcessors(beanFactory)的时候实例化并调用。ConfigurationClassPostProcessor到注册表中&#xff08;internalConfigurationAnnotationProcessor&#xff09;
beanDefs.add(registerPostProcessor(registry, def, CONFIGURATION_ANNOTATION_PROCESSOR_BEAN_NAME));
}
}

3. 思考

在读取配置文件的时候&#xff0c;解析&#xff0c;遇见 context:component-scan&#xff0c;实例化Hander&#xff0c;同时注册各种解析器&#xff0c;调用parse()&#xff0c;开始使用对应的解析器开始解析&#xff0c;componentScanBeanDefinitionParse-parse()&#xff0c;各种解析&#xff0c;最后回注册ConfigurationClassPostProcessor&#xff0c;待到
refresh()->
registerBeanPostProcessors(beanFactory);实例化、调用







推荐阅读
  • 深入解析SpringMVC核心组件:DispatcherServlet的工作原理
    本文详细探讨了SpringMVC的核心组件——DispatcherServlet的运作机制,旨在帮助有一定Java和Spring基础的开发人员理解HTTP请求是如何被映射到Controller并执行的。文章将解答以下问题:1. HTTP请求如何映射到Controller;2. Controller是如何被执行的。 ... [详细]
  • 本文深入探讨了HTTP请求和响应对象的使用,详细介绍了如何通过响应对象向客户端发送数据、处理中文乱码问题以及常见的HTTP状态码。此外,还涵盖了文件下载、请求重定向、请求转发等高级功能。 ... [详细]
  • 2018-2019学年第六周《Java数据结构与算法》学习总结
    本文总结了2018-2019学年第六周在《Java数据结构与算法》课程中的学习内容,重点介绍了非线性数据结构——树的相关知识及其应用。 ... [详细]
  • 深入解析Spring启动过程
    本文详细介绍了Spring框架的启动流程,帮助开发者理解其内部机制。通过具体示例和代码片段,解释了Bean定义、工厂类、读取器以及条件评估等关键概念,使读者能够更全面地掌握Spring的初始化过程。 ... [详细]
  • 深入解析 Android IPC 中的 Messenger 机制
    本文详细介绍了 Android 中基于消息传递的进程间通信(IPC)机制——Messenger。通过实例和源码分析,帮助开发者更好地理解和使用这一高效的通信工具。 ... [详细]
  • 本文详细探讨了HTML表单中GET和POST请求的区别,包括它们的工作原理、数据传输方式、安全性及适用场景。同时,通过实例展示了如何在Servlet中处理这两种请求。 ... [详细]
  • 本文介绍如何使用 Android 的 Canvas 和 View 组件创建一个简单的绘图板应用程序,支持触摸绘画和保存图片功能。 ... [详细]
  • 本文介绍了如何使用JavaScript的Fetch API与Express服务器进行交互,涵盖了GET、POST、PUT和DELETE请求的实现,并展示了如何处理JSON响应。 ... [详细]
  • 本文将探讨2015年RCTF竞赛中的一道PWN题目——shaxian,重点分析其利用Fastbin和堆溢出的技巧。通过详细解析代码流程和漏洞利用过程,帮助读者理解此类题目的破解方法。 ... [详细]
  • yikesnews第11期:微软Office两个0day和一个提权0day
    点击阅读原文可点击链接根据法国大选被黑客干扰,发送了带漏洞的文档Trumps_Attack_on_Syria_English.docx而此漏洞与ESET&FireEy ... [详细]
  • 由二叉树到贪心算法
    二叉树很重要树是数据结构中的重中之重,尤其以各类二叉树为学习的难点。单就面试而言,在 ... [详细]
  • 并发编程 12—— 任务取消与关闭 之 shutdownNow 的局限性
    Java并发编程实践目录并发编程01——ThreadLocal并发编程02——ConcurrentHashMap并发编程03——阻塞队列和生产者-消费者模式并发编程04——闭锁Co ... [详细]
  • SpringMVC RestTemplate的几种请求调用(转)
    SpringMVCRestTemplate的几种请求调用(转),Go语言社区,Golang程序员人脉社 ... [详细]
  • 本文将详细探讨 Java 中提供的不可变集合(如 `Collections.unmodifiableXXX`)和同步集合(如 `Collections.synchronizedXXX`)的实现原理及使用方法,帮助开发者更好地理解和应用这些工具。 ... [详细]
  • 深入理解 JMeter 定时器
    本文详细介绍了JMeter中定时器的功能和使用方法,探讨了其在性能测试中的重要性,并结合实际案例解释了如何合理配置定时器以模拟真实的用户行为。文章还涵盖了定时器的执行顺序及其与其他元件的相互作用。 ... [详细]
author-avatar
手机用户2502870457
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有