@service注解_Spring 中 @Component、@Service 等注解如何被解析?

前言

@Component和@Service都是工作中常用的注解,Spring如何解析?

1.@Component解析流程

找入口

Spring Framework2.0开始,引入可扩展的XML编程机制,该机制要求XML Schema命名空间需要与Handler建立映射关系。

该关系配置在相对于classpath下的/META-INF/spring.handlers中。

711dcd286fe6495a9ce8d516b8e72b30

如上图所示 ContextNamespaceHandler对应context:... 分析的入口。

找核心方法

浏览ContextNamespaceHandler

59bbea41ec1f4872801bc9368ec28344

在parse中有一个很重要的注释

// Actually scan for bean definitions and register them.

ClassPathBeanDefinitionScanner scanner = configureScanner(parserContext, element);

大意是:ClassPathBeanDefinitionScanner#doScan是扫描BeanDefinition并注册的实现。

ClassPathBeanDefinitionScanner 的源码如下:

protected Set doScan(String... basePackages) {   Assert.notEmpty(basePackages, "At least one base package must be specified");   Set beanDefinitions = new LinkedHashSet<>();   for (String basePackage : basePackages) {      //findCandidateComponents 读资源装换为BeanDefinition      Set candidates = findCandidateComponents(basePackage);      for (BeanDefinition candidate : candidates) {         ScopeMetadata scopeMetadata = this.scopeMetadataResolver.resolveScopeMetadata(candidate);         candidate.setScope(scopeMetadata.getScopeName());         String beanName = this.beanNameGenerator.generateBeanName(candidate, this.registry);         if (candidate instanceof AbstractBeanDefinition) {            postProcessBeanDefinition((AbstractBeanDefinition) candidate, beanName);         }         if (candidate instanceof AnnotatedBeanDefinition) {            AnnotationConfigUtils.processCommonDefinitionAnnotations((AnnotatedBeanDefinition) candidate);         }         if (checkCandidate(beanName, candidate)) {            BeanDefinitionHolder definitionHolder = new BeanDefinitionHolder(candidate, beanName);            definitionHolder =                  AnnotationConfigUtils.applyScopedProxyMode(scopeMetadata, definitionHolder, this.registry);            beanDefinitions.add(definitionHolder);            registerBeanDefinition(definitionHolder, this.registry);         }      }   }   return beanDefinitions;}

上边的代码,从方法名,猜测:

findCandidateComponents:从classPath扫描组件,并转换为备选BeanDefinition,也就是要做的解析@Component的核心方法。

概要分析

findCandidateComponents在其父类ClassPathScanningCandidateComponentProvider 中。

public class ClassPathScanningCandidateComponentProvider implements EnvironmentCapable, ResourceLoaderAware {//省略其他代码public Set findCandidateComponents(String basePackage) {   if (this.componentsIndex != null && indexSupportsIncludeFilters()) {      return addCandidateComponentsFromIndex(this.componentsIndex, basePackage);   }   else {      return scanCandidateComponents(basePackage);   }}private Set scanCandidateComponents(String basePackage) {   Set candidates = new LinkedHashSet<>();   try {      String packageSearchPath = ResourcePatternResolver.CLASSPATH_ALL_URL_PREFIX +            resolveBasePackage(basePackage) + '/' + this.resourcePattern;      Resource[] resources = getResourcePatternResolver().getResources(packageSearchPath);        //省略部分代码      for (Resource resource : resources) {        //省略部分代码         if (resource.isReadable()) {            try {               MetadataReader metadataReader = getMetadataReaderFactory().getMetadataReader(resource);               if (isCandidateComponent(metadataReader)) {                  ScannedGenericBeanDefinition sbd = new ScannedGenericBeanDefinition(metadataReader);                  sbd.setSource(resource);                  if (isCandidateComponent(sbd)) {                     candidates.add(sbd);                //省略部分代码      }   }   catch (IOException ex) {//省略部分代码 }   return candidates;}}

findCandidateComponents大体思路如下:

  1. String packageSearchPath = ResourcePatternResolver.CLASSPATH_ALL_URL_PREFIX resolveBasePackage(basePackage) + '/' + this.resourcePattern; 将package转化为ClassLoader类资源搜索路径packageSearchPath,例如:com.wl.spring.boot转化为classpath*:com/wl/spring/boot/**/*.class
  2. Resource[] resources = getResourcePatternResolver().getResources(packageSearchPath); 加载搜索路径下的资源。
  3. isCandidateComponent 判断是否是备选组件
  4. candidates.add(sbd); 添加到返回结果的list

ClassPathScanningCandidateComponentProvider#isCandidateComponent其源码如下:

protected boolean isCandidateComponent(MetadataReader metadataReader) throws IOException {    //省略部分代码   for (TypeFilter tf : this.includeFilters) {      if (tf.match(metadataReader, getMetadataReaderFactory())) {         return isConditionMatch(metadataReader);      }   }   return false;}

includeFilters由registerDefaultFilters()设置初始值,有@Component,没有@Service啊?

protected void registerDefaultFilters() {   this.includeFilters.add(new AnnotationTypeFilter(Component.class));   ClassLoader cl = ClassPathScanningCandidateComponentProvider.class.getClassLoader();   try {      this.includeFilters.add(new AnnotationTypeFilter(            ((Class extends Annotation>) ClassUtils.forName("javax.annotation.ManagedBean", cl)), false));      logger.trace("JSR-250 'javax.annotation.ManagedBean' found and supported for component scanning");   }   catch (ClassNotFoundException ex) {      // JSR-250 1.1 API (as included in Java EE 6) not available - simply skip.   }   try {      this.includeFilters.add(new AnnotationTypeFilter(            ((Class extends Annotation>) ClassUtils.forName("javax.inject.Named", cl)), false));      logger.trace("JSR-330 'javax.inject.Named' annotation found and supported for component scanning");   }   catch (ClassNotFoundException ex) {      // JSR-330 API not available - simply skip.   }}

Spring如何处理@Service的注解的呢????

2.查文档找思路

查阅官方文档,下面这话:

https://docs.spring.io/spring/docs/5.0.17.RELEASE/spring-framework-reference/core.html#beans-meta-annotations

@Component is a generic stereotype for any Spring-managed component. @Repository, @Service, and @Controller are specializations of @Component

大意如下:

@Component是任何Spring管理的组件的通用原型。@Repository、@Service和@Controller是派生自@Component。

@Target({ElementType.TYPE})@Retention(RetentionPolicy.RUNTIME)@Documented// @Service 派生自@Component@Componentpublic @interface Service {   /**    * The value may indicate a suggestion for a logical component name,    * to be turned into a Spring bean in case of an autodetected component.    * @return the suggested component name, if any (or empty String otherwise)    */   @AliasFor(annotation = Component.class)   String value() default "";}

@Component是@Service的元注解,Spring 大概率,在读取@Service,也读取了它的元注解,并将@Service作为@Component处理。

3. 探寻@Component派生性流程

回顾ClassPathScanningCandidateComponentProvider 中的关键的代码片段如下:

private Set scanCandidateComponents(String basePackage) { //省略其他代码 MetadataReader metadataReader                =getMetadataReaderFactory().getMetadataReader(resource);     if(isCandidateComponent(metadataReader)){       //....   }         }public final MetadataReaderFactory getMetadataReaderFactory() {   if (this.metadataReaderFactory == null) {      this.metadataReaderFactory = new CachingMetadataReaderFactory();   }   return this.metadataReaderFactory;}

1. 确定metadataReader

CachingMetadataReaderFactory继承自 SimpleMetadataReaderFactory,就是对SimpleMetadataReaderFactory加了一层缓存。

其内部的SimpleMetadataReaderFactory#getMetadataReader 问:

public class SimpleMetadataReaderFactory implements MetadataReaderFactory {    @Overridepublic MetadataReader getMetadataReader(Resource resource) throws IOException {   return new SimpleMetadataReader(resource, this.resourceLoader.getClassLoader());}    }

这里可以看出

MetadataReader metadataReader =new SimpleMetadataReader(...);

2.查看match方法找重点方法

7284b9f5b8f9435fbb57375200289a77

AnnotationTypeFilter#matchself方法如下:

@Overrideprotected boolean matchSelf(MetadataReader metadataReader) {   AnnotationMetadata metadata = metadataReader.getAnnotationMetadata();   return metadata.hasAnnotation(this.annotationType.getName()) ||         (this.considerMetaAnnotations && metadata.hasMetaAnnotation(this.annotationType.getName()));}

是metadata.hasMetaAnnotation法,从名称看是处理元注解,我们重点关注

逐步分析

找metadata.hasMetaAnnotation

metadata=metadataReader.getAnnotationMetadata();

metadataReader =new SimpleMetadataReader(...)

metadata= new SimpleMetadataReader#getAnnotationMetadata()

//SimpleMetadataReader 的构造方法SimpleMetadataReader(Resource resource, @Nullable ClassLoader classLoader) throws IOException {   InputStream is = new BufferedInputStream(resource.getInputStream());   ClassReader classReader;   try {      classReader = new ClassReader(is);   }   catch (IllegalArgumentException ex) {      throw new NestedIOException("ASM ClassReader failed to parse class file - " +            "probably due to a new Java class file version that isn't supported yet: " + resource, ex);   }   finally {      is.close();   }   AnnotationMetadataReadingVisitor visitor =            new AnnotationMetadataReadingVisitor(classLoader);   classReader.accept(visitor, ClassReader.SKIP_DEBUG);   this.annotationMetadata = visitor;   // (since AnnotationMetadataReadingVisitor extends ClassMetadataReadingVisitor)   this.classMetadata = visitor;   this.resource = resource;}

metadata=new SimpleMetadataReader(...)**.**getAnnotationMetadata()= new AnnotationMetadataReadingVisitor(。。)

也就是说

metadata.hasMetaAnnotation=AnnotationMetadataReadingVisitor#hasMetaAnnotation

其方法如下:

public class AnnotationMetadataReadingVisitor{    // 省略部分代码@Overridepublic boolean hasMetaAnnotation(String metaAnnotationType) {   Collection> allMetaTypes = this.metaAnnotationMap.values();   for (Set metaTypes : allMetaTypes) {      if (metaTypes.contains(metaAnnotationType)) {         return true;      }   }   return false;}}

逻辑很简单,就是判断该注解的元注解在,在不在metaAnnotationMap中,如果在就返回true。

这里面核心就是metaAnnotationMap,搜索AnnotationMetadataReadingVisitor类,没有发现赋值的地方??!。

查找metaAnnotationMap赋值

回到SimpleMetadataReader 的方法,

//这个accept方法,很可疑,在赋值之前执行SimpleMetadataReader(Resource resource, @Nullable ClassLoader classLoader) throws IOException {//省略其他代码AnnotationMetadataReadingVisitor visitor = new AnnotationMetadataReadingVisitor(classLoader);classReader.accept(visitor, ClassReader.SKIP_DEBUG); this.annotationMetadata = visitor; }

发现一个可疑的语句:classReader.accept。

查看accept方法

public class ClassReader {        //省略其他代码public void accept(..省略代码){    //省略其他代码    readElementValues(    classVisitor.visitAnnotation(annotationDescriptor, /* visible = */ true),    currentAnnotationOffset,     true,    charBuffer);}}

查看readElementValues方法

public class ClassReader{    //省略其他代码private int readElementValues(    final AnnotationVisitor annotationVisitor,    final int annotationOffset,    final boolean named,    final char[] charBuffer) {  int currentOffset = annotationOffset;  // Read the num_element_value_pairs field (or num_values field for an array_value).  int numElementValuePairs = readUnsignedShort(currentOffset);  currentOffset += 2;  if (named) {    // Parse the element_value_pairs array.    while (numElementValuePairs-- > 0) {      String elementName = readUTF8(currentOffset, charBuffer);      currentOffset =          readElementValue(annotationVisitor, currentOffset + 2, elementName, charBuffer);    }  } else {    // Parse the array_value array.    while (numElementValuePairs-- > 0) {      currentOffset =          readElementValue(annotationVisitor, currentOffset, /* named = */ null, charBuffer);    }  }  if (annotationVisitor != null) {    annotationVisitor.visitEnd();  }  return currentOffset;}}

这里面的核心就是 annotationVisitor.visitEnd();

确定annotationVisitor

这里的annotationVisitor=AnnotationMetadataReadingVisitor#visitAnnotation

源码如下,注意这里传递了metaAnnotationMap!!

public class AnnotationMetadataReadingVisitor{@Overridepublic AnnotationVisitor visitAnnotation(String desc, boolean visible) {   String className = Type.getType(desc).getClassName();   this.annotationSet.add(className);   return new AnnotationAttributesReadingVisitor(         className, this.attributesMap,              this.metaAnnotationMap, this.classLoader);}}

annotationVisitor=AnnotationAttributesReadingVisitor

查阅annotationVisitor.visitEnd()

annotationVisitor=AnnotationAttributesReadingVisitor#visitEnd()

public class AnnotationAttributesReadingVisitor{@Overridepublic void visitEnd() {   super.visitEnd();   Class extends Annotation> annotationClass = this.attributes.annotationType();   if (annotationClass != null) {      List attributeList = this.attributesMap.get(this.annotationType);      if (attributeList == null) {         this.attributesMap.add(this.annotationType, this.attributes);      }      else {         attributeList.add(0, this.attributes);      }      if (!AnnotationUtils.isInJavaLangAnnotationPackage(annotationClass.getName())) {         try {            Annotation[] metaAnnotations = annotationClass.getAnnotations();            if (!ObjectUtils.isEmpty(metaAnnotations)) {               Set visited = new LinkedHashSet<>();               for (Annotation metaAnnotation : metaAnnotations) {                  recursivelyCollectMetaAnnotations(visited, metaAnnotation);               }               if (!visited.isEmpty()) {                  Set metaAnnotationTypeNames = new LinkedHashSet<>(visited.size());                  for (Annotation ann : visited) {                     metaAnnotationTypeNames.add(ann.annotationType().getName());                  }                  this.metaAnnotationMap.put(annotationClass.getName(), metaAnnotationTypeNames);               }            }         }         catch (Throwable ex) {            if (logger.isDebugEnabled()) {               logger.debug("Failed to introspect meta-annotations on " + annotationClass + ": " + ex);            }         }      }   }}}

内部方法recursivelyCollectMetaAnnotations 递归的读取注解,与注解的元注解(读@Service,再读元注解@Component),并设置到metaAnnotationMap,也就是AnnotationMetadataReadingVisitor 中的metaAnnotationMap中。

总结

大致如下:

ClassPathScanningCandidateComponentProvider#findCandidateComponents

  1. 将package转化为ClassLoader类资源搜索路径packageSearchPath
  2. 加载搜索路径下的资源。
  3. isCandidateComponent 判断是否是备选组件。

内部调用的TypeFilter的match方法:

AnnotationTypeFilter#matchself中metadata.hasMetaAnnotation处理元注解

metadata.hasMetaAnnotation=AnnotationMetadataReadingVisitor#hasMetaAnnotation

就是判断当前注解的元注解在不在metaAnnotationMap中。

AnnotationAttributesReadingVisitor#visitEnd()内部方法recursivelyCollectMetaAnnotations 递归的读取注解,与注解的元注解(读@Service,再读元注解@Component),并设置到metaAnnotationMap

  1. 添加到返回结果的list

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/472432.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

SAP中货源清单创建的几种方法

SAP中采购货源清单创建的方法有以下几种&#xff1a;1、ME01 手动逐个创建。2、ME05 系统自动批量创建。3、通过信息记录中的菜单“附加”&#xff0d;“源清单”创建。4、通过采购合同中的菜单“项目”&#xff0d;“维护货源清单”创建。5、通过计划协议中的菜单“项目”&…

svg载入html,SVG系列教程:SVG简介与嵌入HTML页面的方式

随着技术向前的推进&#xff0c;SVG相关的讨论也越渐频繁。为了紧跟时代的步伐&#xff0c;我也开始步入SVG相关技术的探索与学习之中。从这篇文章开始会在W3cplus站点上陆续向大家推出有关于SVG相关的教程、译文或案例等。感兴趣的同学可以跟随着教程一起了解与学习SVG相关的知…

LeetCode LCP 30. 魔塔游戏(优先队列)

文章目录1. 题目2. 解题1. 题目 小扣当前位于魔塔游戏第一层&#xff0c;共有 N 个房间&#xff0c;编号为 0 ~ N-1。 每个房间的补血道具/怪物对于血量影响记于数组 nums&#xff0c;其中&#xff1a; 正数表示道具补血数值&#xff0c;即血量增加对应数值&#xff1b;负数表…

字符变量赋值规则_Java的常量、变量、数据类型(基础篇二)

标识符标识符&#xff1a;是指在程序中自己定义的内容&#xff0c;如类名、方法名、变量名等等。命名规则&#xff1a;是有硬性要求的关键字&#xff1a;是指Java已经定义好的单词&#xff0c;具有特殊含义&#xff0c;比如public、static、class、void等等标识符可以包含英文字…

python桌面快捷图标_Python创建、删除桌面、启动组快捷方式的例子分享

一、Python创桌面建快捷方式的2个例子 例子一&#xff1a; 代码如下: import os import pythoncom from win32com.shell import shell from win32com.shell import shellcon def createDesktopLnk(filename,lnkname): shortcut pythoncom.CoCreateInstance( shell.CLSID_Shell…

EF---延迟加载技术

EF---延迟加载技术 及时加载延迟加载IQueryabler和IEnumerable优秀&#xff1a;必要的等待&#xff08;拼接sql&#xff09;按需加载 Include 实体状态add,remove,savechanges根据EF包装实体的State进行相应的增删改查操作。如何控制包装实体的状态entry 普能实体与EF实体的区别…

html中li的圆点,CSS中li圆点样式

li {list-style-type:符号名称}css中用list-style-type指定列表(lists)前面符号,如下&#xff1a;li {list-style-type:符号名称}符号名称可用的值为&#xff1a;disc :  CSS1 实心圆circle :  CSS1 空心圆square :  CSS1 实心方块decimal :  CSS1 阿拉伯数字lower-rom…

selenium定位输入框_[Selenium 粗浅笔记] 用Selenium填写表单

要做什么Steiner&#xff1a;[Selenium] 简单介绍​zhuanlan.zhihu.com我们用刚才学到的知识去用selenium来模拟与表单交互&#xff0c;还好有个可以练习的网站 test website 拿这个网站来试试输入用户名&#xff0c;密码ps: 其实这个自己在本地写个网页就行了&#xff0c;ajax…

LeetCode LCP 29. 乐团站位(数学 等差数列)

文章目录1. 题目2. 解题2.1 模拟超时2.2 优化通过1. 题目 某乐团的演出场地可视作 num * num 的二维矩阵 grid&#xff08;左上角坐标为 [0,0])&#xff0c;每个位置站有一位成员。 乐团共有 9 种乐器&#xff0c;乐器编号为 1~9&#xff0c;每位成员持有 1 个乐器。 为保证声…

python三引号注释_python使用三引号来注释的具体原因?

Why didnt python just use the traditional style of comments like C/C/Java uses: /** * Comment lines * More comment lines */ // line comments // line comments // Is there a specific reason for this or is it just arbitrary? 解决方案 Python doesnt use triple…

HiveQL: 数据定义

文章目录1. Hive 数据库2. 修改数据库3. 创建表3.1 管理表3.2 外部表4. 分区表、管理表5. 删除表6. 修改表学习自《Hive编程指南》1. Hive 数据库 create database DBname; hive (default)> show databases; OK default hive Time taken: 0.023 seconds, Fetched: 2 row(s…

can协议crc计算_详解CAN/CAN FD通信中的循环冗余校验(CRC)方法

数据校验是为保证数据的完整性进行的一种验证操作。CAN通信采用CRC校验作为一种重要的错误检测手段,是节点判断CAN帧信息的完整性并产生确认应答的依据。在现场总线通信和控制的实际应用中,工业应用环境往往是极端的温度以及电磁噪声或是其他的恶劣环境,系统在这种条件下能否正…

PHP操作excel类 PHPExcel

PHP操作excel类 PHPExcel http://www.cr173.com/soft/40741.html 我的微云&#xff1a;http://share.weiyun.com/2db79f1438f87999cfb09ca05890d764 下载后&#xff1a; Tests/01simple.php 看代码就可以了 很简单 -------------------- 我的案例(打包)&#xff1a; 将dede…

天玑720支持鸿蒙系统吗,天玑720属于骁龙多少 天玑720处理器相当于骁龙几

天玑700是一款最新推出的5G入门级别的处理器&#xff0c;那么这款手机芯片相当于骁龙多少处理器&#xff1f;处理器性能怎么样&#xff1f;小编为大家带来最新的手机资讯对此感兴趣的小伙伴&#xff0c;快来看看吧。天玑700相当于骁龙多少&#xff1f;在手机的性能方面相当于骁…

html5 职工入职后台管理系统_ChemCMS是一款基于GO+PHP+MYSQL+HTML5构建的化学内容管理系统

ChemCMS是一款基于GOPHPMYSQLHTML5构建的化学内容管理系统&#xff0c;旨在提高化学类企业信息化管理水平&#xff0c;ChemCMS提供了行业所需的库存管理、订单管理、产品管理、客户管理、权限管理全部解决方案&#xff0c;同时我们还提供一体化的在线商城解决方案&#xff0c;大…

oem718d 基准站设置_RTK电台、网络模式作业设置流程

RTK作业的简要流程仪器架设完成基准站和移动站硬件架设与配置&#xff0c;搭建作业的硬件环境。新建工程创建工程&#xff0c;配置参数、坐标等基础信息&#xff0c;完成作业所需的数据基础。求转换参数匹配工程所需平面坐标和默认经纬度坐标&#xff0c;在作业中直接获得所需目…

委托和事件

事件的由来 上文说到委托的安全性不佳&#xff0c;于是我们要将委托本身私有化&#xff0c;但还要暴露若干方法让外界使用。其中最重要的必然就是为委托挂接方法和调用委托&#xff0c;以便间接地调用委托所代表方法。那么事件event关键字就是c#提供给我们的一个语法糖。他并没…

ios html高度自适应,iOS UILabel高度自适应终结篇

释放双眼&#xff0c;带上耳机&#xff0c;听听看~&#xff01;网上大部分的boundingRectWithSize和sizeWithFont 计算出来的宽高在某些有特殊情况下(如链接中有n等等)计算出来的还是有偏差不准&#xff0c;此时用NSAttributedString和label的attributedText计算会迎刃而解1.给…

HiveQL: 数据操作

文章目录1. 向管理表中装载数据2. 通过查询语句向表中插入数据3. 动态分区插入4. 从单个查询语句创建表并加载数据5. 导出数据学习自《Hive编程指南》 1. 向管理表中装载数据 hive (default)> load data local inpath "/home/hadoop/workspace/student.txt">…

formdata.append加多个值_redis的五种数据结构和应用场景:微博微信点赞+加购物车等...

Redis五种数据结构如下&#xff1a;1.String 字符串类型是redis中最基本的数据类型&#xff0c;一个key对应一个value。String类型是二进制安全的&#xff0c;意思是 redis 的 string 可以包含任何数据。如数字&#xff0c;字符串&#xff0c;jpg图片或者序列化的对象。2.Hash …