Spring Bean深入剖析(一)

前言

Spring是分层的 Java SE/EE应用full- stack轻量级开源框架,以IoC(Inversion of Control:反转控制)和AOP( Aspect Oriented Programming:面向切面编程)为内核,提供了展现层 Spring MVC和持久层 Spring JDBC以及业务层事务管理等众多的企业级应用技术,还能整合开源世界众多著名的第三方框架和类库,逐渐成为使用最多的 Java EE企业应用开源框架。

定义spring bean

什么是BeanDefinition?
BeanDefinition是spring Framework中定义Bean的配置元信息接口,包含:

  • Bean的类名,包含包名,即全限定名

  • Bean的行为配置元素,如作用域、自动绑定模式、生命周期回调等

  • 其他Bean引用,又可称作合作者(Collaborators)或者依赖(Dependencies)

  • 配置设置,比如Bean属性(Properties)
    BeanDefinition的元信息
    | 属性(Property) | 说明 |
    | ------------------------ | ---------------------------------------------- |
    | Class | Bean的全类名,必须是具体类,不能是抽象类或接口 |
    | Name | Bean的名称或ID |
    | Scope | Bean的作用域(如Singlton,prototype等) |
    | Constructor arguments | Bean的构造器参数(用于依赖注入) |
    | Properties | Bean的属性设置(用于依赖注入) |
    | Autowiring mode | Bean的自动绑定模式(如:通过名称byName) |
    | Lazy initialization mode | Bean的延迟初始化模块(延迟和非延迟) |
    | Initialization method | Bean的初始化回调方法名称 |
    | Destruction method | Bean销毁回调方法名称 |
    BeanDefinition的构建

  • 通过BeanDefinitionBuilder
    spring源码部分如下:

    /**
     * Programmatic means of constructing
     * {@link org.springframework.beans.factory.config.BeanDefinition BeanDefinitions}
     * using the builder pattern. Intended primarily for use when implementing Spring 2.0
     * {@link org.springframework.beans.factory.xml.NamespaceHandler NamespaceHandlers}.
     *
     * @author Rod Johnson
     * @author Rob Harrop
     * @author Juergen Hoeller
     * @since 2.0
     */
    public final class BeanDefinitionBuilder {
    
      /**
       * Create a new {@code BeanDefinitionBuilder} used to construct a {@link GenericBeanDefinition}.
       */
      public static BeanDefinitionBuilder genericBeanDefinition() {
          return new BeanDefinitionBuilder(new GenericBeanDefinition());
      }
    
      /**
       * Create a new {@code BeanDefinitionBuilder} used to construct a {@link GenericBeanDefinition}.
       * @param beanClassName the class name for the bean that the definition is being created for
       */
      public static BeanDefinitionBuilder genericBeanDefinition(String beanClassName) {
          BeanDefinitionBuilder builder = new BeanDefinitionBuilder(new GenericBeanDefinition());
          builder.beanDefinition.setBeanClassName(beanClassName);
          return builder;
      }
    .........
    

    我们可以通过这个类进行BeanDefinition的构建:

           //通过BeanDefinitionBuilder进行构建
           BeanDefinitionBuilder beanDefinitionBuilder = BeanDefinitionBuilder.genericBeanDefinition(User.class);
           //通过属性设置
           beanDefinitionBuilder.addPropertyValue("name","zhangsan");
           beanDefinitionBuilder.addPropertyValue("id",1L);
           //获取BeanDefinition实例
           BeanDefinition beanDefinition = beanDefinitionBuilder.getBeanDefinition();
    
  • 通过AbstractBeanDefinition以及派生类

        //通过AbstractBeanDefiniton以及派生类
        GenericBeanDefinition genericBeanDefinition = new GenericBeanDefinition();
        //设置Bean的类型
        genericBeanDefinition.setBeanClass(User.class);
        //通过MutablePropertyValues批量操作属性
        MutablePropertyValues propertyValues = new MutablePropertyValues();
        propertyValues.addPropertyValue("name","zhangsan");
        propertyValues.addPropertyValue("id",1L);
        //通过 set MutablePropertyValues 批量操作属性
        genericBeanDefinition.setPropertyValues(propertyValues);

命名spring bean

Bean的名称

  • 每个Bean拥有一个或多个标识符(identifiters),这些标识符在Bean所在的容器必须是唯一的。通常,一个Bean仅有一个标识符,如果需要额外的,可以考虑使用别名(Alias)来扩充。
  • 在基于XML的配置信息中,开发人员可以通过id或者name属性来规定Bean的标识符。通常Bean的标识符由字母组成,允许出现特殊字符。如果要引入Bean的别名的话,可在name属性使用半角逗号,或分号来分隔。
  • Bean的id或name属性并非必须制定,如果留空的话,容器会为Bean自动生成一个唯一的名称。Bean的命名尽管没有限制,不过官方建议使用驼峰命名的方式,更符合Java的命名约定。
    Bean的名称生成器(BeanNameGenrator)
    由Spring framework 2.0.3引入,框架内建两种实现:
  • DefaultBeanNameGenarator:默认通过BeanNameGenarator实现
    BeanNameGenarator的接口定义如下:
    /**
     * Strategy interface for generating bean names for bean definitions.
     *
     * @author Juergen Hoeller
     * @since 2.0.3
     */
    public interface BeanNameGenerator {
      /**
       * Generate a bean name for the given bean definition.
       * @param definition the bean definition to generate a name for
       * @param registry the bean definition registry that the given definition
       * is supposed to be registered with
       * @return the generated bean name
       */
      String generateBeanName(BeanDefinition definition, BeanDefinitionRegistry registry);
    
    }
    
    
    DefaultBeanNameGenarator实现如下:
    **
     * Default implementation of the {@link BeanNameGenerator} interface, delegating to
     * {@link BeanDefinitionReaderUtils#generateBeanName(BeanDefinition, BeanDefinitionRegistry)}.
     *
     * @author Juergen Hoeller
     * @since 2.0.3
     */
    public class DefaultBeanNameGenerator implements BeanNameGenerator {
    
      /**
       * A convenient constant for a default {@code DefaultBeanNameGenerator} instance,
       * as used for {@link AbstractBeanDefinitionReader} setup.
       * @since 5.2
       */
      public static final DefaultBeanNameGenerator INSTANCE = new DefaultBeanNameGenerator();
    
      @Override
      public String generateBeanName(BeanDefinition definition, BeanDefinitionRegistry registry) {
          return BeanDefinitionReaderUtils.generateBeanName(definition, registry);
      }
    }
    
    可以看到这个一个单例模式的实现,但是构造方法并没有私有化,为的是兼容之前的版本。
  • AnnotationBeanNameGenarator:基于注解扫描的BeanNameGenarator实现,起始于Spring framework 2.5
public class AnnotationBeanNameGenerator implements BeanNameGenerator {

    /**
     * A convenient constant for a default {@code AnnotationBeanNameGenerator} instance,
     * as used for component scanning purposes.
     * @since 5.2
     */
    public static final AnnotationBeanNameGenerator INSTANCE = new AnnotationBeanNameGenerator();

    private static final String COMPONENT_ANNOTATION_CLASSNAME = "org.springframework.stereotype.Component";

    private final Map<String, Set<String>> metaAnnotationTypesCache = new ConcurrentHashMap<>();


    @Override
    public String generateBeanName(BeanDefinition definition, BeanDefinitionRegistry registry) {
        if (definition instanceof AnnotatedBeanDefinition) {
            String beanName = determineBeanNameFromAnnotation((AnnotatedBeanDefinition) definition);
            if (StringUtils.hasText(beanName)) {
                // Explicit bean name found.
                return beanName;
            }
        }
        // Fallback: generate a unique default bean name.
        return buildDefaultBeanName(definition, registry);
    }
........

Spring Bean的别名

Bean别名(Alias)的价值

  • 复用现有的BeanDefinition
  • 更具有场景化的命名方法,比如:
    <alias name="myApp-datasource" alias="subSystemA-datasource"/>
    <alias name="myApp-datasource" alias="subSystemB-datasource"/>
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,723评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,003评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,512评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,825评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,874评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,841评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,812评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,582评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,033评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,309评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,450评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,158评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,789评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,409评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,609评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,440评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,357评论 2 352

推荐阅读更多精彩内容