Spring Ioc 源码简单梳理

Spring容器的refresh()【创建刷新】

prepareRefresh()

刷新前的预处理

  1. initPropertySource() 初始化一些属性设置,由子类自定义个性化的属性设置
  2. getEnvironment().validateRequirProperties();检查属性的合法性,如一些必须要有的属性是否存在
  3. earlyApplicationEvents=new LinkedHashSet<ApplicationEvent>();保存容器中的一些早期的事件

obtainFreshBeanFactory()

获取BeanFactory;

  1. refreshBeanFactory();刷新【创建】BeanFactory;
    创建了一个this.beanFactory=new DefaultListableBeanFactory();
    对beanFactory设置序列化ID
    【Spring 5 会优先判断beanFactory是否存在,存在就先进行清空beanFactory包含里面的bean,并调用bean的销毁方法,再重新new一个 DefaultListableBeanFactory对象作为当前beanFactory】
  2. getBeanFactory();返回刚才GenericApplicationContext创建的BeanFactory对象;
  3. 将创建BeanFactory【DefaultListableBeanFactory】返回

prepareBeanFactory(beanFactory);

BeanFactory的预准备工作(BeanFactory进行一些设置)

  1. 设置BeanFactory的类加载器,支持表达式解析器...
  2. 添加部分BeanPostProcessor【ApplicationContextAwareProcessor】
  3. 设置忽略的自动装配的接口EnvironmentAware,EmbeddedValueReslverAware,ResourceLoaderAware,ApplicationEventPublisherAware,MessageSourceAware,ApplicationContextAware,xxx;
  4. 注册可以解析的自动装配,我们能直接在任何组件中自动注入:BeanFactory,ResourceLoader,ApplicationEventPublisher,ApplicationContext
  5. 添加BeanPostProcessor【ApplicationListenerDetector】【ApplicationListenerDetector就是用于将实现ApplicationListener的bean添加到ApplicationContext中】
  6. 添加编译时的AspectJ
  7. 给BeanFactory中注册一些能用的组件:environment【ConfigurableEnvironment】,systemProperties【Map<String,Object>】,systemEnviroment【Map<String,Object>】

postProcessBeanFactory(beanFactory);

BeanFactory准备工作完成后进行的后置处理工作;

  1. 子类通过重新这个方法来在BeanFactory创建并预准备完成以后做进一步的设置

================================以上是BeanFactory的创建以预准备工作============================

invokeBeanFactoryPostProcessors(beanFactory)

执行BeanFactoryProcessor;
BeanFactoryPostProcessor:BeanFactory的后置处理器。在BeanFactory标准初始化之后执行的;
两个接口:BeanFactoryPostProcessor,BeanDefinitionRegistryPostProcessor:

  1. 执行BeanFactoryPostProcessor方法
    先执行BeanDefinitionRegistryPostProcessor:
    1.获取所有的BeanDefinitionRegistryPostProcessor;
    2.看优先级排序PriorityOrdered优先级接口的BeanDefinitionRegistryPostProcessor;
    postProcessor.postProcessBeanDefinitonRegistry(registry)
    3.再执行实现了Ordered顺序接口的BeanDefinitionRegistryPostProcessor;
    postProcessor.postProcessBeanDefinitonRegistry(registry)
    4.最后执行没有实现任何优先级或是顺序接口的BeanDefinitionRegistryPostProcessor;
    postProcessor.postProcessBeanDefinitonRegistry(registry)
    再执行BeanFactoryPostProcessor
    1.获取所有的BeanFactoryPostProcessor;
    2.看优先级排序PriorityOrdered优先级接口的BeanFactoryPostProcessor;
    postProcessor.postPorcessBeanFactory()
    3.再执行实现了Ordered顺序接口的BeanFactoryPostProcessor;
    postProcessor.postPorcessBeanFactory()
    4.最后执行没有实现任何优先级或是顺序接口的BeanFactoryPostProcessor;
    postProcessor.postPorcessBeanFactory()

registerBeanPostProcessors(beanFactory)

注册BeanPostProcessor(Bean的后置处理器)【intercept bean creation.】
不同接口类型的BeanPostProcessor;在Bean创建前后的执行时机是不一样的
BeanPostProcessor,
DestructionAwareBeanPostProcessor,
InstantiationAwareBeanPostProcessor,
SmartInstantiationAwareBeanPostProcessor,
MergedBeanDefinitionPostProcessor【该方法由internalPostProcessors】

  1. 获取所有的BeanPostProcessor;后置处理器都默认可以通过PriorityOrdered,Ordered接口来执行优先级
  2. 先注册PriorityOrdered优先级接口的BeanPostProcessor
    把每一个BeanPostProcessor;添加BeanFactory中
    beanFactory.addBeanPostProcessor(postProcessor);
  3. 再注册Ordered接口
  4. 最后注册没有实现任何优先级接口的
  5. 最终注册MergedBeanDefinitionPostProcessor;
  6. 注册一个ApplicationListenerDetector来在Bean创建完成后检查是否是ApplicationListener,如果是applicationContext.addApplicationListener(ApplicationListener<?>);

initMessageSource()

初始化MessageSource组件(做国际化功能;消息绑定,消息解析);

  1. 获取BeanFactory
  2. 看容器中是否有id为messageSource,类型是MessageSource的组件
    如果有赋值给messageSource,如果没有自己创建一个DelegatingMessageSource;
    MessageSource:取出国际化配置文件中的某个key的值,能按照区域信息获取
  3. 把创建好的MessageSource注册在容器中,以后获取国际化配置文件的时候,可以自动注入MessageSource,然后调用它的getMessage方法实现国际化功能
    beanFactory.registerSingleton(MESSAGE_SOURCE_BEAN_NAME,this.messageSource);

initApplicationEventMulticaster()

初始化事件派发器

  1. 获取BeanFactory
  2. 从BeanFactory中获取applicationEventMulticaster
  3. 如果上一步没有配置;创建一个SimpleApplicationEventMulticaster
  4. 将创建的ApplcaitionEventMulticaster添加到BeanFactory中,以后其他组件自动注入

onResfresh()

留给子容器(子类)

  1. 子类重写这个方法,在容器刷新的时候可以自定义逻辑

regsisterListeners()

给容器中将所有项目里面的ApplicationListener注册进来;

  1. 从容器中拿到所有的ApplicationListener
  2. 将每个监听器添加到事件派发器中
    getApplicationEventMulticaster().addApplicationListenerBean(listener);
  3. 派发之前步骤产生的事件:

finsihBeanFactoryInitialization(beanFactory)

初始化所有剩下的单实例bean

  1. beanFactory.preInstantiateSingletons();初始化后剩下的单实例bean;
    1. 获取容器中的所有bean,依次进行初始化和创建对象
    2. 获取bean的定义信息:RootBeanDefinition
    3. Bean不是抽象的,是单实例的,不是懒加载:
      1. 判断是否是FactoryBean,是否是实现FactoryBean接口的Bean
      2. 不是工厂Bean。利用getBean(beanName);创建对象
        1. getBean(beanName);ioc.getBean();
        2. doGetBean(name,null,null,false);
        3. 先获取缓存中保存的单例Bean。如果能获取到说明这个bean之前被创建过(所有创建过的单实例bean都会被缓存起来)
          从private final Map<String,Object> singletonObjects=new ConcurrentHashMap<>()获取
        4. 缓存中获取不到,开始Bean的创建对象流程
        5. 标记当前bean已经被创建
        6. 获取bean的定义信息
        7. 获取当前bean依赖的其他bean;如果按照getBean()把依赖的bean先创建出来。
        8. 启动单实例bean的创建流程:
          1. createBean(bean,rootBeanDefinition,args);
          2. Object bean=resolveBeforeInstantiation(beanName,mbdToUser);让BeanPostProcess先尝试创建代理对象
            InstantiationAwareBeanPostProcessor:提前执行
            先触发:postProcessBefroeInstantiation();
            如果有返回值:触发postProcessAfterInitialization();
          3. 如果前面的InstantiationAwareBeanPostProcessor没有返回代理对象:
          4. Object beanInstance=doCreateBean(beanName,mbdToUse,args);
            1. 创建Bean实例,createBeanInstance(beanName,rootBeanDefinition,args);
              利用工厂方法或者对象的构造器创建出Bean的实例
            2. applyMergedBeanDefinitionPostProcessor(rootBeanDefinition,beanType,beanName);
              调用MergedBeanDefinitionPostPorcessor的postProcessMergedBeanDefinition方法。
            3. 【Bean属性赋值】populateBean(beanName,beanDefinition,instanceWrapper);
              1.拿到InstantiationAwareBeanPostProcessor后置处理器:postProcessAfterInstantiation();
              2.拿到InstantiationAwareBeanPostProcessor后置处理器:
              postProcessPropertyValues();
              1. 应用Bean属性的值;为属性利用setter方法等进行赋值:applyPropertyValues(beanName,beanDefinition,beanWrapper,propertyValues);
            4. 【Bean初始化】initializeBean(beanName,exposedObject,beanDefinition);
              1. 【执行Aware接口方法】invokeAwareMethods(beanName,bean);执行xxxAware接口方法:BeanNameAware\BeanClassLoaderAware\BeanFactoryAware
              2. 【执行后置处理器】:applyBeanPostProcessorsBeforeInsitalization(wrappedBean,beanName)
                BeanPostProcessor.postProcessBeforeInitiaization()
              3. 【执行初始化方法】invokeInitMethods(beanName,wrappedName,beanDefinition);
                1. 是否是InitializingBean接口实现;执行接口规定的初始化
                2.是否自定义初始化方法;
              4. 【执行后置处理器初始化之后】applyBeanPostProcessorsAfterInitalization
                BeanPostProcessor.postProcessAfterInitiaization()
              5. 注册Beande销毁方法
          5. 将创建的Bean添加到缓存中singletonObjects;
            ioc容器就是这个Map,很多的Map里面保存了单实例Bean,环境信息..
            所有Bean都利用genBean创建完成以后;
            检查所有Bean是否是SmartInitializingSingleton接口;如果是,就执行afterSingletonsInstantiation

finishRefresh()

完成BeanFactory的初始化创建工作,IOC容器就创建完成;

  1. initLifecycleProcessor();初始化和生命周期有关的后置处理器;LifecycleProessor.
    默认从容器中找是否有lifecycleProcessor的组件【LifecycleProessor】;如果没有new DefaultLifecycleProcessor,并加入都容器中。
    实现一个LifecycleProcessor类,beanFactory在此方法回调LifecycleProcessor的onRefresh()方法,当beanFactory销毁时候,会回调onClose()方法;
  2. getLifecycleProcessor().onRefresh();
    拿到前面定义的生命周期处理器(beanFactory);回调onRefresh();
  3. publishEvent(new ContextRefreshedEvent(this));发布容器刷新完成事件
  4. liveBeanView.regsiterApplicationContext(this);

总结

  1. Spring容器在启动的时候,先会保存所有注册进来的Bean定义信息
    1. xml注册bean:<bean>
    2. 注解注册Bean:@Service,@Component,@Bean
  2. Spring容器会合适的时机创建这些Bean
    1. 用到这个bean的时候;利用genBean创建bean;创建好以后保存在容器中;
    2. 统一创建剩下所有的bean的时候;finishBeanFactoryInitialization();
  3. 后置处理器
    每一个bean创建完成,都会使用各种后置处理进行处理;来增强bean的功能;
    AutowiredAnntationBeanPortProcessor:处理自动注入
    AnnotationAwareAspectJAutoProxyCreate:来做AOP功能;
    xxx...
    增强的功能注解
    AsyncAnnotationBeanProcessor
    ......
  4. 事件驱动模型:
    ApplicationListener:事件监听;
    ApplicationEventMulticaster:事件派发;
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,076评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,658评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,732评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,493评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,591评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,598评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,601评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,348评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,797评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,114评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,278评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,953评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,585评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,202评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,442评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,180评论 2 367
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,139评论 2 352

推荐阅读更多精彩内容