zoukankan      html  css  js  c++  java
  • Bean与BeanDefinition的理解——理解spring的基础和关键

    BeanDefinition与Bean的关系, 就好比类与对象的关系. 类在spring的数据结构就是BeanDefinition.根据BeanDefinition得到的对象就是我们需要的Bean.

    我认为理解Bean与BeanDefinition是理解spring的整个架构的基础与关键。我将从Bean与BeanDefinition开始慢慢拨开spring的内幕。

    研究bean与BeanDefinition,我认为可以分为两部分来循序渐进的解析。

    • 一方面从定义入手:研究Bean/BeanDefinition在spring中的定义.
    • 另一方面从动作(操作)入手:研究Bean/BeanDefinition的相关操作。

    BeanDefinition

    BeanDefinition是bean在spring中的描述,有了BeanDefinition我们就可以创建Bean,BeanDefinition是Bean在spring中的定义形态。通过这个 BeanDefinition 定义的数据结构,容器能够方便地对 Bean 进行管理。对于单例的 Bean,在第一次 getBean 的时候读取 BeanDefinition 获取定义,然后再创建和初始化 Bean,并缓存起来。对于原型的 Bean,在每一次 getBean 的时候都需要使用 BeanDefinition 的数据进行创建和初始化,不会缓存 Bean。

    BeanDefinition 在 Spring 内部使用,外部开发者一般感知不到 BeanDefinition 的存在。我们使用时只需要关注通过 xml、注解怎样定义了一个 Bean,然后使用即可。而这中间的解析、创建、初始化都通过 BeanDefinition 来承载。

    接下来我们看看BeanDefinition的相关接口与类.

    定义

    • BeanDefinition接口顶级基础接口,用来描述Bean,里面存放Bean元数据,比如Bean类名、scope、属性、构造函数参数列表、依赖的bean、是否是单例类、是否是懒加载等一些列信息。

    向上
    在这里插入图片描述

    • BeanMetadataElement接口:BeanDefinition元数据,返回该Bean的来源
    • AttributeAccessor接口:提供对BeanDefinition属性操作能力,

    向下
    在这里插入图片描述

    • AbstractBeanDefinition类:抽象类统一实现了BeanDefinition定义的一部分操作,可以说是定义了BeanDefinition很多默认的属性。 正是在AbstractBeanDefinition基础上, Spring衍生出了一些列BeaDefinition。

      这里我们可以关注下重写的equals(),hashcode(), toString()方法

      此外initMethodName属性,destroyMethodName 属性, 这两个属性bean的生命周期有关,此处只提一句,后续讲解。

    接下来。我们看看从AbstractBeanDefinition上衍生出来的几个类

    • RootBeanDefinition:
      代表一个xml,java Config来的BeanDefinition

    • ChildBeanDefinition:
      可以让子BeanDefinition定义拥有从父母哪里继承配置的能力

    • GenericBeanDefinition:
      spring2.5后注册bean首选的是GenericBeanDefinition。GenericBeanDefinition允许动态的设置父bean.GenericBeanDefinition可以作为RootBeanDefinition与ChildBeanDefinition的替代品。

    • AnnotatedBeanDefinition接口
      表示注解类型BeanDefinition。有两个重要的属性,AnnotationMetadata,MethodMetadata分别表示BeanDefinition的注解元信息和方法元信息
      实现了此接口的BeanDefinition可以获取到注解元数据和方法元数据。

    • AnnotatedGenericBeanDefinition类:
      表示@Configuration注解注释的BeanDefinition类

    • ScannedGenericBeanDefinition类:
      表示@Component、@Service、@Controller等注解注释的Bean类

    操作

    动作也可分为两种:
    一种是针对自身的操作: 自提提供给外部的可以操作其本身的动作
    另一种是外部对BeanDefinition的操作

    • BeanDefinitionRegistry接口:具有增,查,删BeanDefinition的能力。一次只能注册一个BeanDefinition.

      实现类SimpleBeanDefinitionRegistry,DefaultListableBeanFactory,GenericApplicationContext等
      一般实现类里都都有一个
      private final Map<String, BeanDefinition> beanDefinitionMap = new ConcurrentHashMap()来存储BeanDefinition.
      在这里插入图片描述

    • BeanDefinitionReader接口: 既可以使用BeanDefinitionRegistry构造。也可以通过loadBeanDefinitions把配置加载为多个BeanDefinition并注册到BeanDefinitionRegistry中。
      可以说是高效版本的BeanDefinitionRegistry.
      实现类有
      XmlBeanDefinitionReader从xml中读取BeanDefinition;
      PropertiesBeanDefinitionReader从Properties文件读取BeanDefinition

    • AnnotatedBeanDefinitionReader类
      对带有注解的BeanDefinition进行注册

    • ClassPathBeanDefinitionScanner类
      可以扫描到@Component @Repository @Service @Controller 的BeanDefinition注册到容器中。

    其他形态

    • BeanDefinitionHolder: BeanDefinition包装类。

    Bean

    Bean是我们需要的对象,是我们从spring内得到的结果,也就是对象实例

    定义

    从定义层面看.Bean其实就是我们需要的对象.

    操作

    我们来看看Bean在spring有哪些操作相关的接口或类。

    • SingletonBeanRegistry接口:与BeanDefinition的注册相应的。Bean的操作也有一个类似的接口来操作Bean.SingletonBeanRegistry接口提供了对Bean的注册,获取,存在性判断等功能。
    • InitializingBean:对于实现 InitializingBean的Bean,它将执行 afterPropertiesSet(); 在所有的 bean 属性被设置之后。
    • InstantiationStrategy:提供 Bean实例化的策略接口,
    • DisposableBean:对于 实现了DisposableBean的Bean ,它将运行 destroy()在 Spring 容器释放该 bean 之后
    • FactoryBean: 生产Bean的Bean.

    其他形态

    • BeanWrapper:
      对Bean的包装.BeanWrapper可以看作是一个从 BeanDefinition 到 Bean 过程中间的产物,可以称为”低级 bean“,在一般情况下,我们不会在实际项目中用到它。BeanWrapper 是 Spring 框架中重要的组件类,它就相当于一个代理类,Spring 委托 BeanWrapper 完成 Bean 属性的填充工作。在 bean 实例被 InstantiatioonStrategy 创建出来后,Spring 容器会将 Bean 实例通过 BeanWrapper 包裹起来,是通过 BeanWrapper.setWrappedInstance() 完成的

    总结:

    BeanDefinition是物料,Bean是成品,理解BeanDefinition与Bean的关系是理解spring的基础

    注:转载于https://www.cnblogs.com/smallstudent/p/11638112.html

  • 相关阅读:
    Jmeter 调试接口用例怎么判断提取的上一个接口返回值是正确的?
    Jmeter 加密处理方法
    Apache Ignite 学习
    jmeter 中 浮点数计算精度问题
    httprunner 使用总结
    oh-my-zsh 安装及使用
    内置装饰器二:@property
    内置装饰器一:@classmethod、@staticmathod
    python实现列表的排序
    Mac git简易使用
  • 原文地址:https://www.cnblogs.com/jing-yi/p/14242016.html
Copyright © 2011-2022 走看看