Spring 的类扫描器分析 - ClassPathBeanDefinitionScanner

本文主要是介绍Spring 的类扫描器分析 - ClassPathBeanDefinitionScanner,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

简介

ClassPathBeanDefinitionScanner继承自ClassPathScanningCandidateComponentProvider,构造时要求指定一个BeanDefinitionRegistry对象,ClassPathBeanDefinitionScanner作用就是将指定包下的类通过一定规则过滤后 将Class 信息包装成 BeanDefinition 的形式注册到IOC容器中,有两个核心的方法:

1、scan方法

public int scan(String... basePackages) {int beanCountAtScanStart = this.registry.getBeanDefinitionCount();doScan(basePackages);// Register annotation config processors, if necessary.if (this.includeAnnotationConfig) {AnnotationConfigUtils.registerAnnotationConfigProcessors(this.registry);}return (this.registry.getBeanDefinitionCount() - beanCountAtScanStart);
}

2、doScan方法:

protected Set<BeanDefinitionHolder> doScan(String... basePackages) {Assert.notEmpty(basePackages, "At least one base package must be specified");Set<BeanDefinitionHolder> beanDefinitions = new LinkedHashSet<>();for (String basePackage : basePackages) {Set<BeanDefinition> 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;
}

自定义扫描器

通过自定义的扫描器,扫描指定包下所有被@MyBeanScanner 注释的类

定义一个注解

@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface MyBeanScanner {
}

编写扫描器

class MyClassPathDefinitonScanner extends ClassPathBeanDefinitionScanner {
private Class type;
public MyClassPathDefinitonScanner(BeanDefinitionRegistry registry, Class<? extends Annotation> type){super(registry,false);this.type = type;
}
public void registerTypeFilter(){addIncludeFilter(new AnnotationTypeFilter(type));
}public void doScan(String basePackage){Set<BeanDefinitionHolder> beanDefinitionHolders = super.doScan(basePackage);beanDefinitionHolders.forEach((item)->{System.out.println(item.getBeanDefinition().getBeanClassName());});
}
}

测试自定义扫描器

 public static void main(String[] args) {AnnotationConfigApplicationContext ac = newAnnotationConfigApplicationContext(AppConfig.class);MyClassPathDefinitonScanner myClassPathDefinitonScanner = new MyClassPathDefinitonScanner(ac, MyBeanScanner.class);myClassPathDefinitonScanner.registerTypeFilter();myClassPathDefinitonScanner.doScan("com.mi.asop.demo.bean");}

Mybatis 的Mapper注册器-ClassPathMapperScanner

Mybatis中的ClassPathMapperScanner继承了ClassPathBeanDefinitionScanner,先调用父类ClassPathBeanDefinitionScanner.doScan方法扫描包路径并注册bd(BeanDefinition对象),再调用processBeanDefinitions处理注册后的bd(BeanDefinition对象),下面是我自己定义的MyMapperClassPathScanner和Mybatis中的基本是一样的:

关于Mybatis的这块内容可参考https://mp.csdn.net/mdeditor/103304489#这篇博客,这里主要模拟mybatis中的ClassPathMapperScanner来完成mapper接口的动态代理对象的扫描和注册,上一篇博客没有利用ClassPathMapperScanner,是手动根据包名遍历解析的。
在这里插入图片描述

MyMapperClassPathScanner

public class MyMapperClassPathScanner extends ClassPathBeanDefinitionScanner {
private MyMapperFactoryBean myMapperFactoryBean = new MyMapperFactoryBean();
public MyMapperClassPathScanner(BeanDefinitionRegistry registry) {super(registry);
}
/***  调用父类的doScan方法* @param basePackages* @return*/
public Set<BeanDefinitionHolder> doScan(String... basePackages) {/*** 调用父类doScan方法拿到扫描到的BeanDefinition*/Set<BeanDefinitionHolder> beanDefinitions = super.doScan(basePackages);if (beanDefinitions.isEmpty()) {System.out.println("No MyBatis mapper was found in '" + Arrays.toString(basePackages) + "' package. Please check your configuration.");} else {this.processBeanDefinitions(beanDefinitions);}return beanDefinitions;
}
/*** 遍历beanDefinitions,动态改变beanClass和给构造方法传值* @param beanDefinitions*/
private void processBeanDefinitions(Set<BeanDefinitionHolder> beanDefinitions){Iterator var3 = beanDefinitions.iterator();while(var3.hasNext()) {BeanDefinitionHolder holder = (BeanDefinitionHolder) var3.next();GenericBeanDefinition definition = (GenericBeanDefinition) holder.getBeanDefinition();System.out.println("Creating MapperFactoryBean with name '" + holder.getBeanName() + "' and '" + definition.getBeanClassName() + "' mapperInterface");definition.getConstructorArgumentValues().addGenericArgumentValue(definition.getBeanClassName());definition.setBeanClass(this.myMapperFactoryBean.getClass());}
}public void registerTypeFilter(){this.addIncludeFilter(new TypeFilter() {public boolean match(MetadataReader metadataReader, MetadataReaderFactory metadataReaderFactory) throws IOException {return true;}});
}protected boolean isCandidateComponent(AnnotatedBeanDefinition beanDefinition) {return beanDefinition.getMetadata().isInterface() && beanDefinition.getMetadata().isIndependent();
}
}

MyImportBeanDefinitionRegister类(同上一篇博客)

第一种注册bd的注释了,主要看第二种,在最下面

public class MyImportBeanDefinitionRegister implements ImportBeanDefinitionRegistrar {
/*
@Override
public void registerBeanDefinitions(AnnotationMetadata annotationMetadata, BeanDefinitionRegistry beanDefinitionRegistry) {Map<String, Object> annotationAttributes = annotationMetadata.getAnnotationAttributes(MyScan.class.getName());String[] basePackages = ( String[])annotationAttributes.get("value");//拿到注解上的扫描的包的名称 com.mi.asop.demo.mapperString packageName =  basePackages[0];//得到包下面所有的类String packagePath = "";String[] names = packageName.split("\\.");for(String name:names){packagePath+=name+"/";}String classPath = this.getClass().getResource("/").getPath()+packagePath;File file = new File(classPath);String[] list = file.list();for(String fileName:list){//com.mi.asop.demo.mapper.userMapperfileName = fileName.substring(0,fileName.lastIndexOf("."));String first = fileName.substring(0, 1);String after = fileName.substring(1);first = first.toLowerCase();String beanName = first+after;BeanDefinitionBuilder builder = BeanDefinitionBuilder.genericBeanDefinition(MyMapperFactoryBean.class);AbstractBeanDefinition beanDefinition = builder.getBeanDefinition();beanDefinition.getConstructorArgumentValues().addGenericArgumentValue(packageName+"."+fileName);beanDefinitionRegistry.registerBeanDefinition(beanName,beanDefinition);}
}
*/
@Override
public void registerBeanDefinitions(AnnotationMetadata annotationMetadata, BeanDefinitionRegistry beanDefinitionRegistry) {Map<String, Object> annotationAttributes = annotationMetadata.getAnnotationAttributes(MyScan.class.getName());String[] basePackages = ( String[])annotationAttributes.get("value");MyMapperClassPathScanner  myMapperClassPathScanner = new MyMapperClassPathScanner(beanDefinitionRegistry);myMapperClassPathScanner.registerTypeFilter();myMapperClassPathScanner.doScan(basePackages[0]);}
}

这篇关于Spring 的类扫描器分析 - ClassPathBeanDefinitionScanner的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:https://blog.csdn.net/u014018804/article/details/103326312
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/713793

相关文章

eclipse如何运行springboot项目

《eclipse如何运行springboot项目》:本文主要介绍eclipse如何运行springboot项目问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目js录当在eclipse启动spring boot项目时出现问题解决办法1.通过cmd命令行2.在ecl

Java中的Closeable接口及常见问题

《Java中的Closeable接口及常见问题》Closeable是Java中的一个标记接口,用于表示可以被关闭的对象,它定义了一个标准的方法来释放对象占用的系统资源,下面给大家介绍Java中的Clo... 目录1. Closeable接口概述2. 主要用途3. 实现类4. 使用方法5. 实现自定义Clos

Linux中的more 和 less区别对比分析

《Linux中的more和less区别对比分析》在Linux/Unix系统中,more和less都是用于分页查看文本文件的命令,但less是more的增强版,功能更强大,:本文主要介绍Linu... 目录1. 基础功能对比2. 常用操作对比less 的操作3. 实际使用示例4. 为什么推荐 less?5.

Jvm sandbox mock机制的实践过程

《Jvmsandboxmock机制的实践过程》:本文主要介绍Jvmsandboxmock机制的实践过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、背景二、定义一个损坏的钟1、 Springboot工程中创建一个Clock类2、 添加一个Controller

SpringBoot实现文件记录日志及日志文件自动归档和压缩

《SpringBoot实现文件记录日志及日志文件自动归档和压缩》Logback是Java日志框架,通过Logger收集日志并经Appender输出至控制台、文件等,SpringBoot配置logbac... 目录1、什么是Logback2、SpringBoot实现文件记录日志,日志文件自动归档和压缩2.1、

MQTT SpringBoot整合实战教程

《MQTTSpringBoot整合实战教程》:本文主要介绍MQTTSpringBoot整合实战教程,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考... 目录MQTT-SpringBoot创建简单 SpringBoot 项目导入必须依赖增加MQTT相关配置编写

spring-gateway filters添加自定义过滤器实现流程分析(可插拔)

《spring-gatewayfilters添加自定义过滤器实现流程分析(可插拔)》:本文主要介绍spring-gatewayfilters添加自定义过滤器实现流程分析(可插拔),本文通过实例图... 目录需求背景需求拆解设计流程及作用域逻辑处理代码逻辑需求背景公司要求,通过公司网络代理访问的请求需要做请

Spring Security介绍及配置实现代码

《SpringSecurity介绍及配置实现代码》SpringSecurity是一个功能强大的Java安全框架,它提供了全面的安全认证(Authentication)和授权(Authorizatio... 目录简介Spring Security配置配置实现代码简介Spring Security是一个功能强

SpringCloud使用Nacos 配置中心实现配置自动刷新功能使用

《SpringCloud使用Nacos配置中心实现配置自动刷新功能使用》SpringCloud项目中使用Nacos作为配置中心可以方便开发及运维人员随时查看配置信息,及配置共享,并且Nacos支持配... 目录前言一、Nacos中集中配置方式?二、使用步骤1.使用$Value 注解2.使用@Configur

Java 中的跨域问题解决方法

《Java中的跨域问题解决方法》跨域问题本质上是浏览器的一种安全机制,与Java本身无关,但Java后端开发者需要理解其来源以便正确解决,下面给大家介绍Java中的跨域问题解决方法,感兴趣的朋友一起... 目录1、Java 中跨域问题的来源1.1. 浏览器同源策略(Same-Origin Policy)1.