Spring IOC和Spring AOP的实现原理(源码主线流程)

2 篇文章 0 订阅
订阅专栏

  正本文参考了《spring技术内幕》和spring 4.0.5源码。本文只描述原理流程的主线部分,其他比如验证,缓存什么可以具体参考源码理解。

Spring IOC

一、容器初始化

      容器的初始化首先是在对应的构造器中进行,在applicationContext的实现类构造器中,首先对参数路径中的${}进行了处理,用系统变量替换(setConfigLocations方法)然后调用refresh方法(这个就是最核心的容器初始化方法)。
ioc容器初始化流程图

1、Resource定位:

      在refresh方法中调用obtainFreshBeanFactory方法告诉子类刷新beanfactory(其中是调用refreshBeanFactory刷新后getBeanFactory获取刷新后的factory返回)。在刷新过程refreshBeanFactory中如果factory已经有了要消除再新建factory,其中loadBeanDefinitions是加载bean定义的方法。

      在loadBeanDefinitions方法中创建了BeanDefinitionReader的实现类调用其loadBeanDefinitions方法(这个方法是重载方法,参数有为Resource的也有为String路径的,getConfigResources方法(默认返回null,子类重写,如ClassPathXmlApplicationContext类)和getConfigLocations方法获得Resource集合和资源路径集合(一般一个为空,一般是将容器的参数path设定为configLocations,ClassPathXmlApplicationContext有一种构造器是不设定configLocations而是直接用参数path生成ClassPathResource集合设定为configResources)分别进行load,实际上以路径为参数的重载方法在定位完Resource也会调用以resource为参数的loadBeanDefinitions来解析载入BeanDefinition,这个是第二步在下面介绍)。

      在BeanDefinitionReader的loadBeanDefinitions(path参数)方法中根据ResourceLoader类型以两种方式加载(如果是ant正则表达式方式的(如PathMatchingResourcePatternResolver)一个路径定位多个resource或者默认方式(applicationContext继承的是DefaultResourceLoader实现方式)定位一个resource),分别调用ResourceLoader的getResource(以/开头的构建ClassPathContextResource,以classpath开头的去掉classpath构建ClassPathResource,如果都不是的尝试构建UrlResource,如果构建失败就调用getResourceByPath这个具体applicationContext实现类里重写的方法构建特定Resource,如FileSystemXmlApplicationContext就是FileSystemResource)或getResources(PathMatchingResourcePatternResolver的正则方式这里不详细描述)完成Resource定位。

2、从Resource中解析和载入BeanDefinition:

      同样在BeanDefinitionReader的loadBeanDefinitions中调用完resourceLoader的getResource获取Resource后将resource作为参数调用自己(BeanDefinitionReader)的loadBeanDefinitions(是一个接口方法给子类实现,因为不同的reader加载resource的方式不同)载入BeanDefinition。

      例如XmlBeanDefinitionReader是对XML文件的IO操作,(将现在要处理的Resource加入当前线程正在处理(ThreadLocal)的Resource集合中)首先从resource中拿出InputStream封装成InputSource调用自身的doLoadBeanDefinitions方法。

      doLoadBeanDefinitions方法中调用doLoadDocument方法封装成Document-----是用validationMode(默认是自动校验方式,意思是如果没有显示定义校验的方式就用XSD方式)和DocumentLoader(XmlBeanDefinitionReader中默认的是DefaultDocumentLoader)等参数调用DocumentLoader的loadDocument方法将Resource封装成Document类(具体封装方式不做介绍,有兴趣的可以自己了解一下,用的是builder模式做的)调用registerBeanDefinitions方法解析载入bean。

      registerBeanDefinitions方法是用BeanDefinitionDocumentReader的registerBeanDefinitions具体解析Document(树形结构,从root(就是beans标签)开始往下解析)中每个element各个标签的解析和载入。其中如果是bean标签BeabDefinitionParserDelegate的parseBeanDefinitionElement方法对XML元素的信息按照spring的bean的规则进行解析(property的解析,当中value和ref解析方式不同,如果是value构建TypedStringValue, 如果ref的话构建RuntimeBeanReference,这个在之后依赖注入的时候用到,还有id,name,等属性的解析)得到的BeanDefinition的封装BeanDefinitionHolder(包括BeanDefinition,beanName(这里是标识符的意思,如果有id,id做标识符,没有id,name属性中第一个别名做标识符)和别名列表(name属性中的内容,如果没有id,name中第一个不作为别名而是标识符))来进行下一步bean的注册(BeanDefinitionReaderUtils.registerBeanDefinition)。

      其他如import,alias等标签自行看源码理解。

3、BeanDefinition在IOC容器的注册

      BeanDefinitionReaderUtils.registerBeanDefinition用BeanDefinitionRegistry(DefaultListableBeanFactory)的registerBeanDefinition方法注册beanName和BeanDefinition(就是把beanName加入到一个已经注册的bean的beanName的Set中,然后put到beanName对应BeanDefinition的map中,其中如果不允许覆盖并且有同名beanName要报错)。再用BeanDefinitionRegistry的registerAlias方法注册beanName和别名列表(put到一个beanName对应alias的map中,其中如果有alias跟beanName相同的要移除)。

二、IOC容器依赖注入
依赖注入

1、getBean第一次调用lazy-init的bean

      是以BeanFactory的getBean方法为入口触发的(实现在AbstractBeanFactory实现类中)。如果是单例会缓存起来只加载一次,如果是FactoryBean这种特殊的bean会把这个bean的实例传入getObjectForBeanInstance方法获得FactoryBean产生的bean(调用FactoryBean的getObject方法,这就是自定义的FactoryBean要重写的方法,AOP也是这个原理,详情见下方AOP分析)。在第一次载入时要先判断这个BeanDefinition在当前BeanFactory有没有,没有就从双亲BeanFactory中找,一直递归。

      找到后要验证是否存在递归依赖,有则报错无则设置当前bean依赖bean的依赖关系到两个map中(一个是被依赖map,一个是依赖map),其中:
      (1)如果是单例第一次载入就调用getSingleton方法(方法中回调了参数中ObjectFactory的getObject方法,这里重写了这个方法调用createBean)获得实例用getObjectForBeanInstance获得FactoryBean产生的bean(如果它是FactoryBean的话)。
      (2)如果是prototype加载调用createBean后调用getObjectForBeanInstance。
      (3)如果是其他scope类型:request、session和global session,这三种就用scope.get获取实例(和getSingleton类似回调重写的getObject也就是调用createBean)后调用getObjectForBeanInstance。

      最后如果getBean指定了requiredType要检验获取的bean能不能转化成指定的类型不能的话就报错。

      createBean方法就是生成bean的方法并对一些比如init-method属性、后置处理器等一些初始化进行了处理。方法中在实例化之前判断是否有post-processor,如果有这样的processor则短路指定bean的创建,直接返回一个proxy而不是指定的bean(这种processor可以指定生成一个其他类型的对象)没有的话用doCreateBean创建bean返回。

      doCreateBean是用createBeanInstance生成BeanWrapper(包装bean)之后用populateBean向其中的bean完成依赖bean的注入(autowire等)。

      createBeanInstance创建beanWrapper时分三类进行处理:
      (1)如果有工厂方法,调用instantiateUsingFactoryMethod创建。
      (2)如果是构造器注入的方式调用autowireConstructor。
      (3)简单方式调用instantiateBean。调用的是策略类(默认SimpleInstantiationStrategy)的instantiate而其中又是通过bean方法是否有跟IOC容器同名的(会被覆盖)来分两类处理(没同名方法的从BeanDefinition中拿出class直接用jdk的反射拿构造器来newinstance一个实例,如果有同名的则是用CGLIB的方式来new一个实例)。

      populateBean为生成的bean依赖注入,先对非简单类型属性有autowire的进行处理,判断这个属性在之前解析载入beanDefinition时property里有没有,有的话进行getBean初始化后放入PropertyValue集合中(这个就是propertyname和value的封装),然后更新依赖map,再对非autowire的或一般属性进行注入,有要转化的要经过valueResolver的转化(如果是RuntimeBeanReference之前载入时XML中配置是ref的就getBean(如果在双亲BeanFactory中就从双亲中取)获得后也放到PropertyValue集合中,也要更新依赖map)。最后再注入到bean中,这里说的注入其实真实发生在最后的BeanWraper的setPropertyValue(propertyValue集合)方法,具体实现就是通过反射的方式获得setter方法赋值。

2、lazy-init==false初始化(只对singleton,也是默认方式)

      在refresh方法中的finishBeanFactoryInitialization方法中进行初始化(实际也是调用getBean方法)。

Spring AOP

      ProxyFacotryBean是FacotryBean的一种实现,FacotryBean要产生bean都要重写getObject方法,而ProxyFacotryBean这里的这个getObject正是为代理做了准备并返回代理对象。首先用initializeAdvisorChain(第一次去取代理对象时初始化一遍)初始化Advisor链后对于singleton和prototype进行区分生成对应的proxy。
aop

1、初始化Advisor链

      initializeAdvisorChain初始化Advisor链是遍历ProxyFacotryBean中配置的interceptorNames,如果结尾有通配符只能是ListableBeanFacotory来加载否则报错,去掉结尾通配符*后调用addGlobalAdvosor(这个是获取ListableBeanFacotory的所有globalAdvisorNames和globalInterceptorNames,分别遍历用getBean(beanName)获取advice,把其中符合通配符格式的advice调用addAdvisorOnChainCreation封装成advicsor后添加到Advisor链,如果结尾没有通配符的情况下无论是singleton还是prototype在获得advice后都要用addAdvisorOnChainCreation方法注册到advisor链上。

    addAdvisorOnChainCreation用namedBeanToAdvisor方法把advice包装成advisor,判断如果advice是单例singleton的话是用AdvisorAdapterRegistry(默认DefaultAdvisorAdapterRegistry单例)wrap方法判断如果这个advice是MethodInterceptor或者AdvisorAdapterRegistry三种固定的adapter(before,afterreturning,throws)如果任一adapter支持的话(支持不支持就是在具体的adapter中判断advice是不是这个adapter对应具体的advice类的子类)就封装成DefaultPointcutAdvisor返回。如果是prototype的话不获取getBean,而是直接用name包装成PrototypePlaceholderAdvisor。

2、生成代理类

      以singleton为例,singleton代理的生成getSingletonInstance方法。是用AopProxyFactory(在构造器中设定了默认的DefaultAopProxyFactory)的createAopProxy方法根据ProxyFacotryBean中配置的target判断是否是个接口(实际上不是这么简单的区分,具体看源码了解)来创建不同AopProxy的子类(JdkDynamicAopProxy或者ObjenesisCglibAopProxy(CglibAopProxy的子类,增加了ObjenesisStd))调用他们各自的getProxy方法以不同的方式创建代理对象返回。

      JdkDynamicAopProxy就是以动态代理的方式构建代理对象返回(具体动态代理原理自行了解哦)。

      CglibAopProxy就是以Cglib的方式进行代理,Cglib采用了非常底层的字节码技术,其原理是通过字节码技术为一个类创建子类,并在子类中采用方法拦截的技术拦截所有父类方法的调用,顺势织入横切逻辑。具体细节超出这文章的范围拉。

      prototype代理的方式大致相同有些许的差别也不做介绍,可以参考源码。

3、调用时拦截

      在调用目标类的方法时因为代理调用的是invoke(jdk动态代理)或者intercept(cglib)。在invoke(jdk动态代理)或者intercept(cglib)中根据目标类被调用方法分别处理。

      如果是hashCode和equals方法直接调用代理类中重写了的hashCode和equals方法(具体参考源码)。
      如果是Adviced接口中定义的方法(ProxyFactoryBean就是Adviced接口实现类)直接以反射的方式拿到method调用方法(AopUtils的invokeJoinpointUsingReflection方法)。
      其他情况就是拿到拦截器链(只初始化一次,每次调用时有个currentInterceptorIndex记录处理到第几个拦截器)调用拦截器的proceed方法前进调用。

      proceed前进调用不是递归,其中用matcher进行匹配,如果匹配上调用拦截器的invoke方法,匹配不上就直接继续前进调用,拦截器interceptor的invoke方法就是通知方法(自己实现的如afterReturning等)对目标方法(实际是拦截器链的proceed前进调用)的具体加强,就是顺序问题等等。

      直到拦截器链前进到底调用target目标类的对应方法(jdk反射获取method调用)。

      初始化拦截器链是通过遍历之前IOC容器getBean获取到advisor链中的Advisor,通过AdvisorAdapterRegistry当中设置的3种adapter(before,afterreturning,throws)的supportsAdvice判断是否支持该advisor,如果支持就将advisor中的advice注册成不同的AdviceInterceptor列表(一个advisor可以被多个adapter支持,因为只要自己写的通知类实现多种advice接口即可)都加入到拦截器链。

loC 和AOP
yfdddong的博客
09-08 175
创建对象。依赖注入。loC是Spring全家桶各个功能模块的基础,创建对象的容器。AOP以loC为基础, AOP是面向切面编程,抽象化的面向对象。AOP
Spring高手之路18——从XML配置角度理解Spring AOP
最新发布
卓越无关环境,保持空杯心态——靡不有初,鲜克有终
05-12 1万+
本文是全面解析面向切面编程的实践指南。通过深入讲解切面、连接点、通知等关键概念,以及通过XML配置实现Spring AOP的步骤。
SpringAOP 原理详解
20年码农
04-14 2048
AOP是一种编程范式,用于在不修改原始代码的情况下向现有应用程序添加新功能。这种编程方式将应用程序分成许多独立的部分,称为切面。这些切面可以在应用程序的不同位置进行编写和维护,从而提高了应用程序的可重用性和可维护性。AOP主要用于实现横切关注点(Cross-Cutting Concerns),例如日志记录、性能监测、事务管理等。通过AOP,我们可以将这些关注点与应用程序的其他部分分离开来,从而使应用程序更加模块化和易于维护。
图文详解 Spring AOP,看完必懂!
良月柒
03-18 2154
程序员的成长之路互联网/程序员/技术/资料共享关注阅读本文大概需要 4 分钟。来自:blog.csdn.net/duxd185120/article/details/109210224学...
Spring AOP 详解
yyhgo_的博客
01-23 6254
Spring AOP 详解
Spring AOP IOC源码笔记.pdf
05-08
1.Spring入门和IOC介绍 2.对象依赖 3.AOP入门 4.JDBCTemplate和Spring事务 5.Spring事务原理 6.Spring事务的一个线程安全问题 7.IOC再回顾和面试题 8.AOP再回顾
spring-IOC-AOP调用大致过程(源码分析)
08-12
spring version: 5.0.0; jdk: 1.8 IOC大致调用顺序(IOC调用的AOP标签解析)
spring ioc aop 源码阅读pdf
04-17
所谓IoC,对于spring框架来说,就是由spring来负责控制对象的生命周期和对象间的关系。这是什么意思呢,举个简单的例子,我们是如何找女朋友的?常见的情况是,我们到处去看哪里有长得漂亮身材又好的mm,然后打听...
基于Java仿Spring实现简单的 IoCAOP【100012181】
05-15
本项目包含了两个不同版本的 IoCAOP 实现,相关源码包结构如下: simple,包含了简单的 IoCAOP 实现代码 IoC,包含了较为复杂的 IoC 实现代码 aop,包含了较为复杂的 AOP 实现代码
拿捏面试官-Spring AOP IOC源码笔记
01-30
Spring AOP IOC源码笔记,看完后轻松拿捏面试官
Spring技术内幕:深入解析 Spring架构与设计原理.pdf
06-19
本书从源代码的角度对Spring的内核和各个主要功能模块的架构、设计和实现原理进行了深入剖析。你不仅能从本书中参透Spring框架的优秀架构和设计思想,还能从Spring优雅的实现源码中一窥Java语言的精髓。本书在开篇之前对Spring的设计理念和整体架构进行了全面的介绍,能让读者从宏观上厘清Spring各个功能模块之间的关系;第一部分详细分析了Spring的核心:IoC容器和AOP的实现,能帮助读者了解Spring的运行机制;第二部分深入阐述了各种基于IoC容器和AOP的Java EE组件在Spring中的实现原理;第三部分讲述了ACEGI安全框架、DM模块以及Flex模块等基于Spring的典型应用的设计与实现。 无论你是Java程序员、Spring开发者,还是平台开发人员、系统架构师,抑或是对开源软件源代码着迷的代码狂人,都能从本书中受益。
SpringAOP详解+快速入门
zhaolk的博客
06-04 2099
本文介绍了 Spring AOP实现原理,并通过实例解析详细阐述了其实现过程。在使用 Spring AOP时,我们需要定义切面类,并为需要实现AOP的方法添加注解。 Spring 会自动将切面逻辑插入到这些方法中,从而实现AOP的功能。
Spring高手之路17——动态代理的艺术与实践
卓越无关环境,保持空杯心态——靡不有初,鲜克有终
04-11 1万+
本文深入分析了JDK和CGLIB两种动态代理技术在Spring框架中的应用。讨论了动态代理的基础概念,通过实例展示了如何实现和应用这两种方法,并比较了它们的性能差异及适用场景。进一步,探讨了在动态代理中实现熔断限流和日志监控的策略,以及如何利用动态代理优化Spring应用的设计和功能。
Spring框架之AOP详解
qq_45515347的博客
09-05 1915
Spring AOP 详解,并实现demo
普歌-云言团队-SpringAOP简介
qq_45812355的博客
07-29 866
什么是AOP AOP 为 Aspect Oriented Programming 的缩写,意思为面向切面编程,是通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。 AOP 是 OOP 的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型。利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。 AOP 的作用及其优势 作用:在程序运行期间,在不修改源码的情况下对方法进行功能增强 优
Spring AOP
Mr.xing的博客
02-21 6034
面向切面
SpringIOCAOP原理
子冉冰清的博客
09-26 2万+
SpringIOCAOP原理 本文讲的是面试之Spring框架IOCAOP实现原理IoC(Inversion of Control) (1). IoC(Inversion of Control)是指容器控制程序对象之间的关系,而不是传统实现中,由程序代码直接操控。控制权由应用代码中转到了外部容器,控制权的转移是所。 IoC(Inversion of Control) (1). IoC(Inversion of Control)是指容器控制程序对象之间的关系,而不是传统实现中,由程序代码直接操控。
深入理解Spring两大特性:IoCAOP
热门推荐
null
02-14 9万+
众所周知,Spring拥有两大特性:IoCAOP。。Spring核心容器的主要组件是Bean工厂(BeanFactory),Bean工厂使用控制反转(IoC)模式来降低程序代码之间的耦合度,并提供了面向切面编程(AOP)的实现。简单来说,Spring是一个轻量级的控制反转(IoC)和面向切面编程(AOP)的容器框架。下面,我们简要说明下这两大特性。
springioc源码
03-26
SpringIoc容器是基于BeanFactory实现的,在BeanFactory的基础上抽象出了ApplicationContenxt接口,是一种更高级别的容器,提供了更多的功能。 在实现IoC容器时,主要使用了以下设计模式: - 工厂模式:...

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

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
写文章

热门文章

  • spring-cloud-gateway之GatewayFilterFactory 1651
  • linux window文件格式转linux文件格式 1372
  • 【转载】 VMware克隆CentOS7 Linux没有eth网卡只有lo 771
  • jstack(查看线程)、jmap(查看内存)和jstat(性能分析) 585
  • VMware虚拟机三种网络模式的区别及配置方法 252

分类专栏

  • java 5篇
  • 大数据
  • 运维 2篇
  • vmware 1篇
  • spring 2篇
  • linux 2篇

最新评论

  • linux window文件格式转linux文件格式

    Zz141221: 加油!!!

  • linux window文件格式转linux文件格式

    Tisfy: 写文章不易,继续加油

大家在看

  • AI写真教程:Stable Diffusion 之 IPAdapter-FaceId,打造独一无二的AI人物画像
  • 数据生成 | Matlab实现基于K-means和SVM的GMM高斯混合分布的数据生成
  • 计算机SCI期刊,中科院2区TOP,收稿范围广泛! 417
  • 世界奇观短视频制作,AI加持,新手也能月入上万
  • AJAX中get和post的区别

最新文章

  • spring-cloud-gateway之GatewayFilterFactory
  • linux window文件格式转linux文件格式
  • VMware虚拟机三种网络模式的区别及配置方法
2020年1篇
2019年9篇
2018年1篇

目录

目录

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值

深圳SEO优化公司双龙优化大运网站制作设计横岗网站优化软件盐田网站搜索优化深圳seo网站优化横岗SEO按效果付费深圳模板推广龙华营销网站广州百搜词包坪山建站大鹏seo优化广州网站建设设计沙井百搜标王永湖百姓网标王推广广州百姓网标王推广惠州SEO按天收费塘坑网站优化按天计费沙井网站优化推广坪地百度网站优化排名坑梓模板制作观澜网站建设松岗建网站石岩seo网站优化大运seo排名永湖网站制作福永SEO按天收费深圳高端网站设计罗湖网站推广工具龙岗高端网站设计福永网站改版歼20紧急升空逼退外机英媒称团队夜以继日筹划王妃复出草木蔓发 春山在望成都发生巨响 当地回应60岁老人炒菠菜未焯水致肾病恶化男子涉嫌走私被判11年却一天牢没坐劳斯莱斯右转逼停直行车网传落水者说“没让你救”系谣言广东通报13岁男孩性侵女童不予立案贵州小伙回应在美国卖三蹦子火了淀粉肠小王子日销售额涨超10倍有个姐真把千机伞做出来了近3万元金手镯仅含足金十克呼北高速交通事故已致14人死亡杨洋拄拐现身医院国产伟哥去年销售近13亿男子给前妻转账 现任妻子起诉要回新基金只募集到26元还是员工自购男孩疑遭霸凌 家长讨说法被踢出群充个话费竟沦为间接洗钱工具新的一天从800个哈欠开始单亲妈妈陷入热恋 14岁儿子报警#春分立蛋大挑战#中国投资客涌入日本东京买房两大学生合买彩票中奖一人不认账新加坡主帅:唯一目标击败中国队月嫂回应掌掴婴儿是在赶虫子19岁小伙救下5人后溺亡 多方发声清明节放假3天调休1天张家界的山上“长”满了韩国人?开封王婆为何火了主播靠辱骂母亲走红被批捕封号代拍被何赛飞拿着魔杖追着打阿根廷将发行1万与2万面值的纸币库克现身上海为江西彩礼“减负”的“试婚人”因自嘲式简历走红的教授更新简介殡仪馆花卉高于市场价3倍还重复用网友称在豆瓣酱里吃出老鼠头315晚会后胖东来又人满为患了网友建议重庆地铁不准乘客携带菜筐特朗普谈“凯特王妃P图照”罗斯否认插足凯特王妃婚姻青海通报栏杆断裂小学生跌落住进ICU恒大被罚41.75亿到底怎么缴湖南一县政协主席疑涉刑案被控制茶百道就改标签日期致歉王树国3次鞠躬告别西交大师生张立群任西安交通大学校长杨倩无缘巴黎奥运

深圳SEO优化公司 XML地图 TXT地图 虚拟主机 SEO 网站制作 网站优化