专栏首页求道Spring:没有人比我更懂Java对象的创建!

Spring:没有人比我更懂Java对象的创建!

有道无术,术尚可求也!有术无道,止于术!

一、开篇一问

一个Spring Bean是Java对象吗?那么一个对象是Spring Bean吗?

带着这个问题我们一起来回顾Spring的生命周期流程,彻底了解一个类在Spring中究竟做了哪些操作!

我们先看一段代码!

image-20200922221511683

诚然,相信只要是使用过Spring的人都应该知道,从Spring容器中获取的对象,里面的EmailServiceImpl是被Spring自动注入了的!

但是相同的点是,他们两个都是一个Java对象,都是加载在JVM里面的,那么至少我们现在可以回答的问题了:

「SpringBean一定是一个java对象;但是一个Java对象却不一定是一个SpringBean!」

那么我们是不是可以这样认为,只有被Spring管理的类,才能够称之为一个Bean,其他的都称之为对象!那么Spring是如何将一个类从一个普通的类转换为一个Spring Bean的呢?他究竟是经过了哪些步骤呢?我们一起探究一下!

二、Spring生命周期的大胆猜测

这里分享一个阅读源码的小技巧:捉大放小,连蒙带猜!8字真言,我们在阅读源码过程中,因为你要知道,每一个被开源出来的优秀框架,其源码的体系都是极其庞大复杂的,我们不能面面俱到,所以在看源码过程中一定不能被细枝末节缠住,一定要先理清楚整个框架的一个大致思想和大致的框架体系,再去搞那些细枝末节,其效率会好很多,其次在看源码过程中,我们一定要大胆的去想,去猜测,如果这个功能让你自己去写,你会怎么实现!

我们今天学习SpringBean的生命周期也是按照这个8字真言去学习,通过我们之前所学,Spring大致有以下的功能:

  1. 他会帮我们自动的创建对象然后保存起来!
  2. 他会帮我们完成属性的填充!
  3. 如果我们设置了Aop的功能,他会帮我们自动的代理,实现切面功能!

我们从平常的使用中,至少可以得知以上的三点,如果让你自己去实现,必会如何实现呢?

  • 首先他既然能够帮我们自己创建对象,那么他肯定是通过反射来创建的,通过反射来创建,就必定绕不过去要使用Class对象创建,那么我们如何获取Class对象呢?去扫描项目,将指定的包下的加了注解的类文件切割获取Class名称,通过反射加载Class名称,反射创建java对象!
  • 我们要完成属性的填充,为了方便和性能方面,我肯定会把这些创建好的对象保存起来,无疑Map容器是最合适的!
  • 我们在创建一个对象完成之后,反射拿到里面的属性,如果需要填充,我们先去我们之前保存的容器里面去取,取不出来在反射吧这个依赖的属性创建出来,然后填充进对象再保存在容器里面,从而完成了属性的注入!
  • 填充完成属性之后,我们拿当前对象,去与Aop逻辑进行对比,判断是否需要代理,不需要则创建完成,保存进Map容器,需要代理则对当前这个类进行jdk或者cglib的代理然后再保存进容器里面!

「于是乎,我们自己实现了一个Spring管理一个Bean的所有过程,画个图,他大概长这样!」

image-20200922230911796

自己实现看起来,整个流程就很清晰,扫描、创建、注入、代理、保存一应俱全,但是Spring的实现方式远比我们自己实现的要复杂的多得多!

三、Spring的生命周期流程

Spring作者希望,Spring在着手管理一个Bean的时候,它希望能够让Spring的使用者能够插手,Spring把一个类对象变成一个Java Object的每一步,怎么理解呢?

比如我们买了一栋新房子,这个房子需要去装修,你自己去装修诚然不够专业,不能够面面俱到,所以是我们就找了一个装修公司帮助我们装修新房,于是装修公司就开始预先画好的图纸进行装修,但是在装修的过程中,你为了让自己的新家更加温馨,你想挂一些壁画在墙上,但是图纸上却没有!于是你就找装修公司,要求装修公司在新家的墙上挂上一些壁画!装修公司在接受到你的请求之后,就吩咐装修的工人在图纸之外去给你在墙上挂上壁画之后,然后再接着装修!

上面这个小故事有 这样几个角色,我们把它和Spring对照起来!

  • 你:代表框架的使用者!
  • 新房:代表一个Class文件,你自己也能够装修,但是不够专业,所以交给装修公司! 那么你自己创建对象可能某些使用用起来很麻烦,所以我们交给了Spring容器!
  • 装修公司:代表着Spring容器!
  • 图纸:代表预设步骤,Spring原本就存在的步骤!
  • 工人:Spring提供的各种接口!我们可以通过Spring工厂提供的接口做各种自定义的配置!

上面的小故事,大致可以描述Spring生命周期的核心思想!Spring再对一个Class文件实例化成具体的Spring Bean的时候,它提供了各种接口,由我们自己实现!然后再实例化过程中,不同的时机,去调用不同的接口!从而完成Spring的整个生命周期的创建!

Spring的生命周期大致分为以下部分!

  1. 扫描项目,将项目指定目录下的Class文件转换为Class对象!
  2. 读取Class对象属性包装为BeanDefinition,然后保存在一个Map中!(不难理解,他是为了后续创建或者读取这个类的信息更加方便取而创立的)
  3. 将全部的类转化为 BeanDefinition 并保存之后,开始调用第一个回调接口BeanFactoryPostProcessor#postProcessBeanFactory()!
    • 「它的调用时机是将扫描到的Class文件转换为 BeanDefinition 之后调用的,我们可以通过回调的方法获取所有的BeanDefinition ,而后续的所有对Class的操作都是基于BeanDefinition 操作的,所以,我们可以通过修改它,来改变后续的流程!」
  4. 先从当前的容器对象取当前要创建的对象,当取出来的对象为null时开始着手创建对象!
  5. 做一系列的验证,比如验证这个类是否被排除、是否正在创建中、是否有依赖Bean【@DependsOn】注解、是否是单例等等!
  6. 验证通过之后,开始通过反射创建这个对象!
  7. 合并BeanDefinition ,这里涉及到Spring之前版本使用的父子容器的概念,属于另外一个知识点不做讲解!
  8. 判断当前对象是不是单例、是不是支持循环引用、是不是正在创建等!
  9. 执行第二个接口回调InstantiationAwareBeanPostProcessor#postProcessAfterInstantiation()方法!
    • 「它的执行时机是实例化完成之后,属性填充之前,它的返回值是一个布尔值,当返回false时,不做自动属性填充!」
  10. 执行第三个接口回调InstantiationAwareBeanPostProcessor#postProcessProperties()方法!
    • 「他的执行时机是,实例化之后,属性填充检查之后,属性填充之前!它会返回一个属性,后续的属性填充会使用这个方法返回的值!我们可以在这个方法里面修改对应Bean的注入的值!」
  11. 填充属性到对象!
  12. 调用第四个回调接口BeanNameAware#setBeanName()方法!
    • 「调用时机:属性填充给完毕后,调用初始化方法之前;它的功能是能获取bean的Name!」
  13. 调用第五个回调接口BeanClassLoaderAware#setBeanClassLoader()
    • 「调用时机:BeanNameAware之后,他的功能是传入bean的类加载器;」
  14. 调用第六个回调接口BeanFactoryAware#setBeanFactory()!
    • 「调用时机:BeanClassLoaderAware之后,用于设置beanFactory!」
  15. 调用第七个回调接口BeanPostProcessor#postProcessBeforeInitialization()方法
    • 「调用时机是部分Aware之后,初始化方法之前!传入当前实例化好的对象和beanName,再初始化前做修改!」
  16. 回调第八个比较重要的生命周期的初始化方法,它可以是一个InitializingBean接口的bean,也可以是xml中配置的类,也可以是被加了@PostConstruct注解的方法!
    • 「该方法内部逻辑可以用户自己编写,调用时机为:实例化完成之后调用!」
  17. 回调第九个回调接口 BeanPostProcessor#postProcessAfterInitialization()方法!
    • 「该方法的调用时机为初始化方法执行之后,这里也是Bean实例化后的最后一步,也是SpringAop实现的重要的一步!」
  18. 注册销毁方法,以便Spring容器销毁的时候进行方法的销毁!

整体的方法流程示例图如下:

image-20200923004654968

四、对应源码结构图

本文分享自微信公众号 - JAVA程序狗(javacxg),作者:皇甫嗷嗷叫

原文出处及转载信息见文内详细说明,如有侵权,请联系 yunjia_community@tencent.com 删除。

原始发表时间:2020-09-24

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • 一个Spring Bean从诞生到逝去的九次人生转折!

    关于Spring生命周期的学习,前面已经写过很多篇文章去不断的探究Spring对一个Bean的创建、管理过程,在整个SpringBean的生命周期中,BeanP...

    止术
  • 牛逼哄哄的Spring是怎么被MyBatis给征服了?

    其实前几篇文章已经写了好多有关于Spring源码的文章,事实上,很多同学虽然一直在跟着阅读、学习这些Spring的源码教程,但是一直都很迷茫,这些Spring的...

    止术
  • 想要学会Spring源码,你必知必会的BeanDefinition原理!

    这是一个经典的面试题,什么是java Object?万物皆对象,在Java内部所有的类,经过创建之后都可以称之为一个对象,SpringBean也是一个java ...

    止术
  • SpringBoot v2.4.0 正式发布,支持Java15

    第一时间收到了GitHub推送的SpringBoot发版邮件,2020.11.13日正式发布了v2.4.0,这是2.4.x分支的第一个版本。

    恒宇少年
  • 了解 spring 的整体架构

    POJO其实是比javabean更纯净的简单类或接口。 POJO严格地遵守简单对象的概念,而一些JavaBean中往往会封装一些简单逻辑。 POJO主要用于...

    lvgo
  • Spring模块化设计:Spring功能特性如何在不同模块中组织?

    Spring的模块化设计和Java 9 的模块化设计不是对等的,不过Spring 的模块化确实可以运用到Java 9 的模块化里面去 ,那是为什么呢,那是因为在...

    极客小智
  • 为什么说 Java 程序员到了必须掌握 Spring Boot 的时候?

    Spring Boot 2.0 的推出又激起了一阵学习 Spring Boot 热,就单从我个人的博客的访问量大幅增加就可以感受到大家对学习 Spring Bo...

    技术zhai
  • Spring实战1:Spring初探

    现在的Java程序员赶上了好时候。在将近20年的历史中,Java的发展历经沉浮。尽管有很多为人诟病的产品,例如applets、EJB、Java Data Obj...

    java架构师
  • [译]2020年Spring状态报告

    近日VMware发布了2020年Spring状态报告,该报告调查了1000多位不同行业的springboot开发者、架构师、技术经理,以了解企业当前如何使用Sp...

    东溪陈姓少年
  • 从新手到行家:Spring、Spring Boot\Cloud 学习指南 | 视频版

    作为一名 Java 程序员,现在你已经很难离开 Spring 了,Spring 历经十几年发展事实上已经成为 Java 领域最重要的框架(没有之一),从早期的 ...

    纯洁的微笑

扫码关注云+社区

领取腾讯云代金券