就想搞明白,component-scan 是怎么把Bean都注册到Spring容器的!

一、前言

忒复杂,没等搞明白大促都过去了!

你经历过618双11吗?你加入过大促时候那么多复杂的营销活动赚几毛钱吗?你开发过连读明白玩法都需要一周但只使用3天的大促需求吗?有时候对于有些产品的需求真的是太复杂了,复杂到开发、测试都需要在整个过程中不断的学习最后才可能读懂产品为啥这样的玩,要是一个长期的活动可能也就算了,培养用户心智吗!但这一整套拉新、助力、激活、下单、投保、领券、消费、开红包等等一连串的骚操作下来,如果在线上只用3天呢,或者是只用1天,那TM连参与的用户都没弄明白呢,活动就结束了,最后能打来什么样好的数据呢?对于这样流程复杂,估计连羊毛当都看不上!!!

以上只是举个例子,大部分时候并不会搞的这么恶心,评审也是过不去的!而同样的道理用在程序设计开发和使用中也是一样的,如果你把你的代码逻辑实现的过于分散,让外部调用方在使用的时候,需要调用你的接口多个和多次,还没有消息触达,只能定时自己轮训你的接口查看订单状态,每次还只能查10条,查多了你说不行,等等反人类的设计,都会给调用方带来要干你的体会。

所以,如果我们能在完成目的的情况下,都是希望尽可能流程简单、模式清晰、自动服务。那这在Spring的框架中也是有所体现的,这个框架的普及使用程度和它所能带来的方便性是分不开的,而我们如果能做到如此的方便,那肯定是一种好的设计和实现。

二、目标

其实到本章节我们已经把关于 IOC 和 AOP 全部核心内容都已经实现完成了,只不过在使用上还有点像早期的 Spring 版本,需要一个一个在 spring.xml 中进行配置。这与实际的目前使用的 Spring 框架还是有蛮大的差别,而这种差别其实都是在核心功能逻辑之上建设的在更少的配置下,做到更简化的使用。

这其中就包括:包的扫描注册、注解配置的使用、占位符属性的填充等等,而我们的目标就是在目前的核心逻辑上填充一些自动化的功能,让大家可以学习到这部分的设计和实现,从中体会到一些关于代码逻辑的实现过程,总结一些编码经验。

三、方案

首先我们要考虑🤔,为了可以简化 Bean 对象的配置,让整个 Bean 对象的注册都是自动扫描的,那么基本需要的元素包括:扫描路径入口、XML解析扫描信息、给需要扫描的Bean对象做注解标记、扫描Class对象摘取Bean注册的基本信息,组装注册信息、注册成Bean对象。那么在这些条件元素的支撑下,就可以实现出通过自定义注解和配置扫描路径的情况下,完成 Bean 对象的注册。除此之外再顺带解决一个配置中占位符属性的知识点,比如可以通过 ${token} 给 Bean 对象注入进去属性信息,那么这个操作需要用到 BeanFactoryPostProcessor,因为它可以处理 在所有的 BeanDefinition 加载完成后,实例化 Bean 对象之前,提供修改 BeanDefinition 属性的机制 而实现这部分内容是为了后续把此类内容结合到自动化配置处理中。整体设计结构如下图:

结合bean的生命周期,包扫描只不过是扫描特定注解的类,提取类的相关信息组装成BeanDefinition注册到容器中。

在XmlBeanDefinitionReader中解析<context:component-scan />标签,扫描类组装BeanDefinition然后注册到容器中的操作在ClassPathBeanDefinitionScanner#doScan中实现。

  • 自动扫描注册主要是扫描添加了自定义注解的类,在xml加载过程中提取类的信息,组装 BeanDefinition 注册到 Spring 容器中。
  • 所以我们会用到 <context:component-scan /> 配置包路径并在 XmlBeanDefinitionReader 解析并做相应的处理。这里的处理会包括对类的扫描、获取注解信息等
  • 最后还包括了一部分关于 BeanFactoryPostProcessor 的使用,因为我们需要完成对占位符配置信息的加载,所以需要使用到 BeanFactoryPostProcessor 在所有的 BeanDefinition 加载完成后,实例化 Bean 对象之前,修改 BeanDefinition 的属性信息。这一部分的实现也为后续处理关于占位符配置到注解上做准备

四、实现

1. 工程结构

small-spring-step-13
└── src
    ├── main
    │   └── java
    │       └── cn.bugstack.springframework
    │           ├── aop
    │           │   ├── aspectj
    │           │   │   └── AspectJExpressionPointcut.java
    │           │   │   └── AspectJExpressionPointcutAdvisor.java
    │           │   ├── framework 
    │           │   │   ├── adapter
    │           │   │   │   └── MethodBeforeAdviceInterceptor.java
    │           │   │   ├── autoproxy
    │           │   │   │   └── MethodBeforeAdviceInterceptor.java
    │           │   │   ├── AopProxy.java
    │           │   │   ├── Cglib2AopProxy.java
    │           │   │   ├── JdkDynamicAopProxy.java
    │           │   │   ├── ProxyFactory.java
    │           │   │   └── ReflectiveMethodInvocation.java
    │           │   ├── AdvisedSupport.java
    │           │   ├── Advisor.java
    │           │   ├── BeforeAdvice.java
    │           │   ├── ClassFilter.java
    │           │   ├── MethodBeforeAdvice.java
    │           │   ├── MethodMatcher.java
    │           │   ├── Pointcut.java
    │           │   ├── PointcutAdvisor.java
    │           │   └── TargetSource.java
    │           ├── beans
    │           │   ├── factory
    │           │   │   ├── config
    │           │   │   │   ├── AutowireCapableBeanFactory.java
    │           │   │   │   ├── BeanDefinition.java
    │           │   │   │   ├── BeanFactoryPostProcessor.java
    │          
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: component-scan报错是指在Spring配置文件中使用了component-scan标签扫描组件时出现了错误。可能的原因包括: 1. 扫描路径配置错误:component-scan标签中的base-package属性配置错误,导致无法扫描到需要的组件。 2. 组件类定义错误:扫描到的组件类定义不符合Spring的要求,例如缺少必要的注解或配置。 3. 依赖注入错误:扫描到的组件类中存在依赖注入错误,例如注入的bean不存在或注入方式不正确。 解决方法包括: 1. 检查component-scan标签中的base-package属性是否正确配置。 2. 检查组件类定义是否符合Spring的要求,例如是否添加了必要的注解或配置。 3. 检查依赖注入是否正确,例如注入的bean是否存在或注入方式是否正确。 4. 查看错误信息,根据错误信息进行排查和解决。 ### 回答2: component-scan是Spring框架中用来自动扫描并注册Bean的组件之一。它通常会自动扫描指定路径下的所有类,并将标记有特定注解的类注册Spring Bean。 如果在配置文件中配置了component-scan,但是在启动时出现了报错,可能有以下几种原因: 1. 配置文件中component-scan的路径错误 如果component-scan扫描的路径错误,Spring框架就会找不到需要注册Bean,从而报错。需要检查路径是否正确,并且确保扫描路径下确实有需要注册Bean的类存在。 2. 需要注册Bean类缺少必要的注解 component-scan通常会查找标记有特定注解的类,并将其注册Spring Bean。如果需要注册Bean类缺少必要的注解,那么在扫描并注册Bean的时候就会报错。需要检查需要注册Bean类是否标记了正确的注解,比如@Service、@Component、@Controller、@Repository等。 3. 版本不兼容 有时候,新版本的Spring框架与旧版本的Spring配置文件不兼容,也会导致component-scan报错。需要检查Spring框架和配置文件的版本是否匹配。如果不匹配,需要更新版本或者修改配置文件。 4. xml命名空间冲突 如果配置文件中引入了多个命名空间,有时候会出现命名空间冲突的情况,导致component-scan无法正常工作。需要检查配置文件中是否存在命名空间冲突的问题,并及时进行修改。 总之,在出现component-scan报错时,需要认真检查配置文件和代码,找出错误的原因,并进行修正。只有这样才能使我们的Spring应用程序正常工作,达到我们预期的效果。 ### 回答3: component-scan 报错是 Spring 框架中常见的问题,通常是因为 Spring 无法找到需要扫描的包或组件的原因。下面我将讨论一些可能导致 component-scan 报错的常见原因,以及如何解决这些问题。 1. 扫描的包路径错误 在配置 component-scan 的时候,指定的扫描包路径可能不正确。解决方法是检查 applicationContext.xml 配置文件中 component-scan 标签内指定的包路径是否正确,是否存在这个路径下。正确的格式为: < context:component-scan base-package="com.example"/> 如果你的包路径不在这个路径下,需要对标签中的包路径进行更改。 2. 在路径中排除了类 有时候在扫描的包路径中排除了类,比如排除了 controller 包,但是在 controller 包中定义了一些组件。解决方法是检查在 component-scan 标签上下文中的过滤器配置是否正确,确保所有需要的类都被扫描到了。 < context:component-scan base-package="com.example" exclude-filter="org.springframework.stereotype.Controller"/> 3. 缺少 Bean 的配置信息 有时候会出现 component-scan 不起作用的问题,原因是缺少 bean 的配置信息。解决方法是在 applicationContext.xml 文件中手动添加 bean 的配置信息,确保所有需要的 bean 被正确配置。 < bean id="exampleBean" class="com.example.ExampleBean"/> 4. 所需的 jar 包未加载 在使用 Spring 框架时,通常需要加载相关的 jar 包,否则 component-scan 报错时应检查相关 jar 包是否被正确加载,并且与版本兼容。 上述是在我遇到 component-scan 报错时常见的原因和解决方法,希望对你有所帮助。如果以上解决方法均不能解决问题,则需要检查应用程序的其它方面是否存在问题,如是否正确配置了数据库连接,或是否存在其它错误。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值