欢迎来到皮皮网网首页

【empi 源码】【android搜题源码】【影片小程序源码】spring源码例子

来源:认证小程序源码 时间:2024-12-28 21:08:39

1.Spring源码系列-BeanPostProcessor与BeanFactoryPostProcessor
2.Spring源码从入门到精通---@Import(五)
3.76 张图,源码剖析 Spring AOP 源码,源码小白居然也能看懂,源码大神,源码请收下我的源码膝盖!
4.Spring Cloud OpenFeign源码FeignClientFactoryBean原理
5.谈谈Spring中的源码empi 源码NoSuchBeanDefinitionException源码
6.25. Spring源码篇之SpEL表达式

spring源码例子

Spring源码系列-BeanPostProcessor与BeanFactoryPostProcessor

       在Spring框架中,BeanPostProcessor与BeanFactoryPostProcessor各自承担着不同的源码职责,它们在IoC容器的源码工作流程中起着关键作用。

       BeanFactoryPostProcessor作用于BeanDefinition阶段,源码对容器中Bean的源码定义进行处理。这个过程发生在BeanFactory初始化时,源码对BeanDefinition进行修改或增强,源码提供了一种在不修改源代码的源码情况下定制Bean的机制。相比之下,源码BeanPostProcessor则在Bean实例化之后生效,源码对已经创建的Bean对象进行进一步处理或替换,提供了更晚、更灵活的扩展点。

       以制造杯子为例,BeanFactoryPostProcessor相当于在选择材料和形状阶段进行定制,而BeanPostProcessor则在杯子制造完成后,进行诸如加花纹、抛光等深加工。

       在Spring框架中,BeanPostProcessor的使用场景较为广泛,尤其在实现AOP(面向切面编程)时,通过使用代理类替换原始Bean,实现如日志记录、事务管理等功能。

       此外,容器在启动后,还会进行消息源初始化、广播器初始化及监听器初始化,为Bean实例化做好准备。完成这些准备工作后,容器会调用registerBeanPostProcessors方法注册BeanPostProcessor,对已创建的Bean进行进一步处理。同时,初始化消息源、广播器和监听器,为后续事件处理做好基础。

       总结,BeanFactoryPostProcessor与BeanPostProcessor在Spring IoC容器中的作用各有侧重。前者侧重于对BeanDefinition的定制,后者则是在Bean实例化后的进一步加工,两者共同为构建灵活、可扩展的IoC容器提供了强大的支持。

       在深入分析Spring框架的android搜题源码源码时,我们发现refresh()方法的实现中包含了对BeanFactoryPostProcessor和BeanPostProcessor的注册与处理。这些处理步骤确保了容器能够在启动时对Bean进行正确的配置和初始化。

       文章中通过一个例子展示了如何使用BeanFactoryPostProcessor替换已注册Bean的实现,以及对其源码的分析。通过例子和源码的结合,读者能够更直观地理解这些后置处理器在Spring框架中的应用和工作原理。

Spring源码从入门到精通---@Import(五)

       深入解析如何给容器注册bean

       通过ComponentScan+注解如@Controller,@Service,@Compoment,@Repository实现自动扫描bean

       @Bean+@Configuration定义导入第三方bean

       利用@Import快速批量导入组件,优势在于简化配置

       文章重点解析@Import的三种用法:直接导入容器、自定义importSelector实现、自定义ImportBeanDefinitionRegistrar手动注册

       1)@import注解直接导入容器,id默认为全类名

       2) 自定义importSelector类,返回需要注册的全类名数组

       3) 实现ImportBeanDefinitionRegistrar接口,自定义组件注册和id

       通过@Import源码,导入的实质是一个数组,允许批量导入多个类

       演示通过import将组件如color和red导入容器,并展示容器中组件的打印

       提供JUnit测试类,重复利用方法提取getDefinitionNames(),简化测试步骤

       新增1)@Import基础使用部分,删除原有代码,便于理解@Import

       运行示例,展示导入组件后的容器打印结果,突出import的优势

       详细步骤:

       2)自定义myImportSelector类实现ImportSelector,返回新增组件路径,结合扫描自定义类

       结果展示:blue和yellow组件成功注册容器,验证自定义importSelect功能

       3)实现ImportBeanDefinitionRegistrar接口,自定义组件名注册到容器

       junit测试不变,运行结果:验证容器中包含red、yellow组件,满足自定义id需求

张图,剖析 Spring AOP 源码,小白居然也能看懂,大神,请收下我的膝盖!

       本文将简要介绍AOP(面向切面编程)的基础知识与使用方法,并深入剖析Spring AOP源码。首先,我们需要理解AOP的基本概念。

       1. **基础知识

**

       1.1 **什么是AOP?

**

       AOP全称为Aspect Oriented Programming,即面向切面编程。AOP的思想中,周边功能(如性能统计、日志记录、事务管理等)被定义为切面,核心功能与切面功能独立开发,然后将两者“编织”在一起,这就是AOP的核心。

       AOP能够将与业务无关、影片小程序源码却为业务模块共同调用的逻辑封装,减少系统重复代码,降低模块间的耦合度,有利于系统的可扩展性和可维护性。

       1.2 **AOP基础概念

**

       解释较为官方,以下用“方言”解释:AOP包括五种通知分类。

       1.3 **AOP简单示例

**

       创建`Louzai`类,添加`LouzaiAspect`切面,并在`applicationContext.xml`中配置。程序入口处添加`"睡觉"`方法并添加前置和后置通知。接下来,我们将探讨Spring内部如何实现这一过程。

       1.4 **Spring AOP工作流程

**

       为了便于理解后面的源码,我们将整体介绍源码执行流程。整个Spring AOP源码分为三块,结合示例进行讲解。

       第一块是前置处理,创建`Louzai`Bean前,遍历所有切面信息并存储在缓存中。第二块是后置处理,创建`Louzai`Bean时,主要处理两件事。第三块是执行切面,通过“责任链+递归”执行切面。

       2. **源码解读

**

       注意:Spring版本为5.2..RELEASE,否则代码可能不同!这里,我们将从原理部分开始,逐步深入源码。

       2.1 **代码入口

**

       从`getBean()`函数开始,进入创建Bean的逻辑。

       2.2 **前置处理

**

       主要任务是遍历切面信息并存储。

       这是重点!请务必注意!获取切面信息流程结束,后续操作都从缓存`advisorsCache`获取。

       2.2.1 **判断是否为切面

**

       执行逻辑为:判断是否包含切面信息。

       2.2.2 **获取切面列表

**

       进入`getAdvice()`,生成切面信息。

       2.3 **后置处理

**

       主要从缓存拿切面,与`Louzai`方法匹配,创建AOP代理对象。

       进入`doCreateBean()`,执行后续逻辑。

       2.3.1 **获取切面

**

       首先,查看如何获取`Louzai`的切面列表。

       进入`buildAspectJAdvisors()`,女神节网站源码方法用于存储切面信息至缓存`advisorsCache`。随后回到`findEligibleAdvisors()`,从缓存获取所有切面信息。

       2.3.2 **创建代理对象

**

       有了`Louzai`的切面列表,开始创建AOP代理对象。

       这是重点!请仔细阅读!这里有两种创建AOP代理对象方式,我们选择使用Cglib。

       2.4 **切面执行

**

       通过“责任链+递归”执行切面与方法。

       这部分逻辑非常复杂!接下来是“执行切面”最核心的逻辑,简述设计思路。

       2.4.1 **第一次递归

**

       数组第一个对象执行`invoke()`,参数为`CglibMethodInvocation`。

       执行完毕后,继续执行`CglibMethodInvocation`的`process()`。

       2.4.2 **第二次递归

**

       数组第二个对象执行`invoke()`。

       2.4.3 **第三次递归

**

       数组第三个对象执行`invoke()`。

       执行完毕,退出递归,查看`invokeJoinpoint()`执行逻辑,即执行主方法。回到第三次递归入口,继续执行后续切面。

       切面执行逻辑已演示,直接查看执行方法。

       流程结束时,依次退出递归。

       2.4.4 **设计思路

**

       这部分代码研究了大半天,因为这里不是纯粹的责任链模式。

       纯粹的责任链模式中,对象内部有一个自身的`next`对象,执行当前对象方法后,启动`next`对象执行,直至最后一个`next`对象执行完毕,或中途因条件中断执行,责任链退出。

       这里`CglibMethodInvocation`对象内部无`next`对象,通过`interceptorsAndDynamicMethodMatchers`数组控制执行顺序,依次执行数组中的对象,直至最后一个对象执行完毕,责任链退出。

       这属于责任链,实现方式不同,后续会详细剖析。下面讨论类之间的java共享锁源码关系。

       主对象为`CglibMethodInvocation`,继承于`ReflectiveMethodInvocation`,`process()`的核心逻辑在`ReflectiveMethodInvocation`中。

       `ReflectiveMethodInvocation`的`process()`控制整个责任链的执行。

       `ReflectiveMethodInvocation`的`process()`方法中,包含一个长度为3的数组`interceptorsAndDynamicMethodMatchers`,存储了3个对象,分别为`ExposeInvocationInterceptor`、`MethodBeforeAdviceInterceptor`、`AfterReturningAdviceInterceptor`。

       注意!这3个对象都继承了`MethodInterceptor`接口。

       每次`invoke()`调用时,都会执行`CglibMethodInvocation`的`process()`。

       是否有些困惑?别着急,我将再次帮你梳理。

       对象与方法的关系:

       可能有同学疑惑,`invoke()`的参数为`MethodInvocation`,没错!但`CglibMethodInvocation`也继承了`MethodInvocation`,可自行查看。

       执行逻辑:

       设计巧妙之处在于,纯粹的责任链模式中,`next`对象需要保证类型一致。但这里3个对象内部没有`next`成员,不能直接使用责任链模式。怎么办呢?就单独设计了`CglibMethodInvocation.process()`,通过无限递归`process()`实现责任链逻辑。

       这就是我们为什么要研究源码,学习优秀的设计思路!

       3. **总结

**

       本文首先介绍了AOP的基本概念与原理,通过示例展示了AOP的应用。之后深入剖析了Spring AOP源码,分为三部分。

       本文是Spring源码解析的第三篇,感觉是难度较大的一篇。图解代码花费了6个小时,整个过程都沉浸在代码的解析中。

       难度不在于抠图,而是“切面执行”的设计思路,即使流程能走通,将设计思想总结并清晰表达给读者,需要极大的耐心与理解能力。

       今天的源码解析到此结束,有关Spring源码的学习,大家还想了解哪些内容,欢迎留言给楼仔。

Spring Cloud OpenFeign源码FeignClientFactoryBean原理

       Spring Cloud OpenFeign的FeignClientFactoryBean在实例化过程中,通过FactoryBean接口实现,GetObject方法的关键步骤包括获取FeignContext、配置Feign.Builder、创建HardCodedTarget和调用loadBalance方法。这些步骤涉及自动配置、FeignClientSpecification的使用、Logger和Builder组件的定制以及动态代理的生成。最后,getObject方法返回的是一个接口的代理类,用于执行远程调用。

       详细分析:

       FeignClientFactoryBean在Spring容器中,通过getObject方法转化为实际的FeignClient实例。首先,它从FeignContext获取相关配置,这个配置在引入OpenFeign依赖时自动注入。接下来,通过getTarget方法,FeignClientFactoryBean配置了Builder组件,如Logger(非Slf4j)、RequestInterceptor、Encoder和Decoder等,同时考虑了用户自定义组件的配置。之后,创建了HardCodedTarget,基于FeignClient接口、注解值和完整URL构建,然后通过loadBalance方法,整合了LoadBalancerFeignClient和HystrixTargeter,进行负载均衡和目标URL定位。

       在newInstance方法中,解析了接口方法的注解,生成了MethodHandler,并用FeignInvocationHandler封装,这个InvocationHandler在代理类实例化时被调用,实现了远程调用。最终,通过Proxy.newProxyInstance动态生成了代理类,完成FeignClientFactoryBean的实例化过程。

       总的来说,FeignClientFactoryBean实例化是通过一系列配置和代理生成,实现了Spring Cloud OpenFeign的远程调用功能。如果你对源码的深入理解感兴趣,下期文章将继续解析调用源码细节。

谈谈Spring中的NoSuchBeanDefinitionException源码

       组织 spring 框架的基本知识后,我们容易发现NoSuchBeanDefinitionException 是常见问题,多数 spring 开发者都经历过。本文将深入讨论NoSuchBeanDefinitionException,包括异常概念,常见触发场景以及相应解决策略。具体内容以 JavaConfig 为例。

       无命名 Bean 定义异常

       此异常名言直译为 “找不到指定的 Bean 定义”,其 Java 文档说明了此异常在 spring 容器中找不到符合特定 bean 定义的情况被抛出。常见引起此异常的情况是 spring 管理的上下文中未能找到指定 Bean。下面我们将具体分析几种触发场景。

       依赖注入时未找到 bean

       在某个 BeanA 声明需要注入一个名为 BeanB 的 Bean 时,如果 spring 上下文中未定义 BeanB,就会抛出 NoSuchBeanDefinitionException。这种情况下,可能是 BeanB 未使用 spring 注解(如@Component、@Repository等)注释,或者对应的包未被 spring 扫描到。要解决这个问题,检查 BeanB 是否已标记为 spring bean,并确认对应的包是否已被包含在扫描范围内。

       多个候选 bean

       另一种常见情形是 spring 容器中存在多个类型相同但无法识别的候选 Bean,比如定义了两个实现 IBaneB 的类(BeanB1 和 BeanB2)。此时,依赖注入时没有具体指示,导致异常抛出。此时可通过附加 @Qualifier 注解来指定具体的候选 Bean,或者使用 @Primary 标记一个,spring 将选择它进行注入。

       通过名称获取非存在 bean

       若要通过名称获取一个尚未定义的 bean 也会引发 NoSuchBeanDefinitionException。确认所请求的名称对应的 bean 是否已经在 spring 上下文中定义。

       代理 Bean 与实现

       spring 的 AOP 机制借助代理提供功能扩展,其中代理类和目标类或实现类在实现或接口层次上的不同会导致异常。当使用接口进行依赖注入时,spring 容器能正常识别和管理。但使用真实的类进行注入时,容器可能会遇到问题,因为代理类并未继承或实现注入的类。在 spring 的事务管理场景中,通过接口注入可以避免此异常。

       总结,解决 NoSuchBeanDefinitionException 需要从 spring 上下文的 bean 定义、扫描范围和注入方式着手,确保 spring 环境的配置与 bean 的定义完全吻合,合理使用 spring 的特性可以有效避免或解决这类异常。

. Spring源码篇之SpEL表达式

       Spring的SpEL表达式,即Spring Expression Language,是Spring框架中实现复杂功能的关键组件。在Spring中,独立的spring-expression模块用于支持这一功能。本文将提供对SpEL表达式源码的简要分析,以帮助理解其基本用法。

       在AbstractBeanFactory中,有一个名为beanExpressionResolver的属性,用于配置默认的表达式解析器。在初始化BeanFactory时,通过AbstractApplicationContext#prepareBeanFactory设置默认值,该值默认为开启状态,可通过配置参数spring.spel.ignore=false来关闭表达式功能。

       核心解析组件是BeanExpressionResolver,它提供了evaluate方法,用于解析传入的表达式并返回结果。作为实现类,StandardBeanExpressionResolver具体实现evaluate方法,执行解析任务。

       解析SpEL表达式的接口是ExpressionParser,它接收表达式和ParserContext,后者定义了解析规则。关键子类包括SpelExpressionParser、InternalSpelExpressionParser和TemplateAwareExpressionParser。在解析过程中,会调用TemplateAwareExpressionParser#parseExpressions方法,该方法进一步调用InternalSpelExpressionParser#doParseExpression,实现表达式的详细解析。解析流程的关键步骤是tokenizer.process和eatExpression方法,它们负责识别和处理特殊字符以及逻辑运算。

       SpEL表达式本质上是一个语法树结构,涉及复杂的运算、对象访问和方法调用。它支持的字符规范包括括号、逻辑运算符(如or、and)、比较运算符(如>、<)、点号(用于访问对象属性)、问号(用于条件判断)、美元符号(用于访问变量)等。

       以下是使用SpEL表达式的简单示例:

       案例一

       输出特定值或表达式的结果。

       案例二

       对数据集进行处理,例如筛选、排序或计算。

       案例三

       执行对象方法,如调用实例方法或访问静态方法。

       案例四

       使用SpEL获取Spring容器中的Bean实例,包括使用@和&注解来分别获取普通Bean和FactoryBean。

       通过以上分析,我们大致了解了SpEL表达式的功能和基本用法。理解这些关键类及其功能有助于在实际开发中灵活运用SpEL,提高代码的可维护性和可读性。尽管SpEL的实现细节复杂,掌握其核心概念和用法足以应对常见的应用场景。

6. Spring源码篇之FactoryBean

       FactoryBean是Spring提供的一个功能强大的小型工厂,用于灵活创建所需Bean。在框架与Spring整合时,尤其是Mybatis-plus中,通过注解可以自动生成Spring Bean,而FactoryBean的功能正是实现批量动态生成Bean。下面详细介绍FactoryBean的源码解析。

       首先,我们来看看如何判断一个对象是否为FactoryBean。在Spring的实例化过程中,如果类实现了FactoryBean接口,则会被识别为FactoryBean。而获取FactoryBean时,通常在Bean名称前加上"&"符号。

       接下来,我们深入分析FactoryBean的接口。

       FactoryBean接口定义了如何创建Bean,包含两个主要方法:getObject和isInstance。getObject用于返回创建的Bean实例,isInstance用于判断一个对象是否由FactoryBean创建。

       SmartFactoryBean是FactoryBean的子接口,它提供了额外的特性,允许决定是否提前实例化对象。

       在实际使用中,FactoryBean的实例化过程较为关键。如果不希望立即实例化某个非懒加载单例Bean,则需要确保它未被识别为FactoryBean。例如,UserBean的实例化代码在正常情况下不会打印任何输出,表明并未实例化。而通过将UserBean实现为SmartFactoryBean,并使isEagerInit返回true,就能在控制台中观察到UserBean的实例化过程。

       获取FactoryBean创建的Bean有多种方式。通过在Bean名称前加"&",可以获取到由getObject方法生成的Bean。此外,若需要获取FactoryBean本身,则可以使用多个"&"符号,Spring会循环遍历,直至获取到实际的Bean。

       在Spring实例化完成后,通常会调用getObjectForBeanInstance方法来获取真正的Bean实例。这一过程包括了共享实例(sharedInstance)的引用和Bean名称的处理。最终,通过调用getObject方法,我们能够获取到由FactoryBean生成的实际Bean。

       以Mybatis-plus中的MapperFactoryBean为例,说明了如何在实际项目中应用FactoryBean。MapperFactoryBean是Mybatis-plus提供的一个FactoryBean,用于自动注册Mapper接口为Spring Bean。

       总结而言,FactoryBean在Spring中扮演着灵活创建和管理Bean的重要角色,尤其在需要动态生成或自定义Bean创建逻辑的场景中。通过理解其源码和使用方法,开发者可以更高效地整合各类框架与Spring,实现更为灵活和高效的系统构建。

Spring 源码学习 :initMessageSource

       前言

       阅读完registerBeanPostProcessors源码后,接下来就是initMessageSource这一步骤,其主要功能是初始化国际化文件。

       按照惯例,首先通过官网了解国际化的用法,然后深入研究源码。

       官网1..1. Internationalization using MessageSource[1]中提到,MessageSource的主要作用是使用国际化,定制不同的消息。

       需要注意的是,MessageSource定义的Bean名称必须为messageSource,如果找不到则会默认注册DelegatingMessageSource作为messageSource的Bean。

       1. 创建国际化文件

       2. 声明MessageSource

       在JavaConfig中声明MessageSource,记得名字一定要叫做messageSource!

       3. 测试结果

       执行后输出结果如下:

       了解了国际化是如何使用的之后,再看看这一步的源码,就知道其作用了!

       initMessageSource源码

       这块源码唯一值得关注的地方就是,Bean的名称必须要是messageSource。

       总结

       本文通过官网,了解到什么是国际化,以及国际化的使用,并结合代码和源码,知其然,知其所以然。

       当然本文需要注意的地方就是国际化MessageSource的Bean名称要必须为messageSource。