Spring BeanDefinition使用介绍
作者:程序员小潘 发布时间:2023-11-24 10:29:10
1. 前言
Spring最重要的一个概念当属Bean了,我们写的Controller、Service、Dao凡是加了对应注解交给Spring管理的,都是Spring容器中的一个Bean。把我们自己写的类变成一个Bean交给Spring管理有很多的好处,比如我们不用自己去new对象了,Spring会帮我们在适当的时候去实例化对象;再比如Spring还会维护Bean的依赖关系,自动注入依赖的对象;再比如Spring还可以基于Bean生成代理对象,来对原生的Bean进行功能上的增强等等。
在Spring源码中,有一个重要的接口BeanDefinition
,它用来表示Bean的定义。比如Bean的名称、所属的Class、是否需要延迟实例化,是单例还是原型?等等信息。不管你是通过xml来配置,还是JavaConfig的方式,还是基于注解的方式,在Spring看来它们都是一个BeanDefinition
。
2. BeanDefinition
BeanDefinition是一个接口,本身的类图还是比较简单的,继承了AttributeAccessor
和BeanMetadataElement
接口。
2.1 AttributeAccessor
AttributeAccessor接口代表一个属性访问器,它的作用是可以给Bean附加一些自定义的属性,并提供访问方法。
例如:originalTargetClass
属性代表AOP被代理的原始Class对象;preserveTargetClass
属性代表是否基于类进行代理。
AttributeAccessorSupport是其中一个实现类,它底层通过LinkedHashMap
来存储这些附加属性。
public interface AttributeAccessor {
void setAttribute(String name, @Nullable Object value);
Object getAttribute(String name);
Object removeAttribute(String name);
boolean hasAttribute(String name);
String[] attributeNames();
}
2.2 BeanMetadataElement
BeanMetadataElement接口我也不是很清楚具体作用是啥,网上对它的介绍也比较少。
public interface BeanMetadataElement {
/**
* 对于常规Bean,可以获得Bean对应的Resource
*/
@Nullable
Object getSource();
}
经过实验发现,对于常规的Bean来说,可以通过getSource()
方法获取Bean对应的资源。例如:现在有一个BeanDefinition它的BeanClass来自ClassPath下的包,getSource()方法可以获取到这个Class文件对应的FileSystemResource资源。
2.3 BeanDefinition
终于看到主角了,BeanDefinition还是一个接口,它提供了一系列方法来读写Bean的一些属性定义,方法名基本都能见名知意,注释里已经把方法的作用写上了。
public interface BeanDefinition extends AttributeAccessor, BeanMetadataElement {
/**
* 父Bean的name 对应XML里的 <bean parent=""> 配置
*/
void setParentName(@Nullable String parentName);
String getParentName();
/**
* BeanClassName 对应XML里的 <bean class=""> 配置
*/
void setBeanClassName(@Nullable String beanClassName);
String getBeanClassName();
/**
* Bean的作用域 单例/原型
*/
void setScope(@Nullable String scope);
String getScope();
/**
* 是否延迟加载
* 默认为fale,容器启动时自动实例化单例Bean
* 设置为true,则在getBean时再实例化
*/
void setLazyInit(boolean lazyInit);
boolean isLazyInit();
/**
* 依赖的其它Bean
* BeanFactory保证先实例化依赖的Bean
*/
void setDependsOn(@Nullable String... dependsOn);
String[] getDependsOn();
/**
* 当前Bean是否作为其它Bean依赖注入时的候选Bean,默认为true
* Spring根据Type注入时,如果存在多个实现类,会抛出异常
* 通过将autowireCandidate设置为false,该Bean将不再是依赖注入时的候选Bean
* 根据name注入时,不受影响
*/
void setAutowireCandidate(boolean autowireCandidate);
boolean isAutowireCandidate();
/**
* 依赖注入 存在多个候选Bean时,优先注入primary=true的
*/
void setPrimary(boolean primary);
boolean isPrimary();
void setFactoryBeanName(@Nullable String factoryBeanName);
String getFactoryBeanName();
void setFactoryMethodName(@Nullable String factoryMethodName);
String getFactoryMethodName();
/**
* Bean的构造函数参数值
*/
ConstructorArgumentValues getConstructorArgumentValues();
default boolean hasConstructorArgumentValues() {
return !getConstructorArgumentValues().isEmpty();
}
/**
* 获取Bean实例的属性值
*/
MutablePropertyValues getPropertyValues();
default boolean hasPropertyValues() {
return !getPropertyValues().isEmpty();
}
/**
* Bean初始化的方法名
*/
void setInitMethodName(@Nullable String initMethodName);
@Nullable
String getInitMethodName();
/**
* Bean销毁的方法名
*/
void setDestroyMethodName(@Nullable String destroyMethodName);
String getDestroyMethodName();
void setRole(int role);
int getRole();
/**
* Bean的描述
*/
void setDescription(@Nullable String description);
String getDescription();
/**
* 是否是单例的
*/
boolean isSingleton();
/**
* 是否是原型的
*/
boolean isPrototype();
/**
* 是否是抽象Bean,如果是将不会实例化
*/
boolean isAbstract();
@Nullable
String getResourceDescription();
/**
* 获取原始的BeanDefinition
*/
@Nullable
BeanDefinition getOriginatingBeanDefinition();
}
2.4 AbstractBeanDefinition
AbstractBeanDefinition是一个抽象类,实现了绝大多数BeanDefinition的功能,源码也很简单,通过大量属性来记录BeanDefinition对应方法设置的属性值,代码就不贴了。
2.5 GenericBeanDefinition
GenericBeanDefinition是一个类,它是BeanDefinition的标准实现,继承自AbstractBeanDefinition。刚刚已经说过了,由于AbstractBeanDefinition已经实现了绝大多数方法了,所以GenericBeanDefinition的代码也极其简单,只实现了一个读写parentName
的方法。
2.6 AnnotatedGenericBeanDefinition
顾名思义,AnnotatedGenericBeanDefinition是一个基于注解的BeanDefinition标准实现,它继承自GenericBeanDefinition,已经获得了BeanDefinition完整的能力了。只不过,由于是基于注解的标准实现,所以它还实现了AnnotatedBeanDefinition
接口,以此来获得获取注解元数据的能力。
public interface AnnotatedBeanDefinition extends BeanDefinition {
// 获取注解元数据
AnnotationMetadata getMetadata();
// 获取工厂方法元数据的能力
MethodMetadata getFactoryMethodMetadata();
}
AnnotatedGenericBeanDefinition的代码也同样极其简单,只是提供了两个属性metadata
和factoryMethodMetadata
来记录注解元数据和工厂方法元数据而已。
2.7 其它实现
本文着重分析了AnnotatedGenericBeanDefinition这一条分支,实际上在Spring中BeanDefinition还有很多实现类,例如:
RootBeanDefinition:ChildBeanDefinition:基于父Bean的子BeanDefinition。
ConfigurationClassBeanDefinition:加了@Bean注解的BeanDefinition。
ScannedGenericBeanDefinition:加了@Component及其派生注解的BeanDefinition。
来源:https://javap.blog.csdn.net/article/details/128352190


猜你喜欢
- 默认spring只在发生未被捕获的runtimeexcetpion时才回滚。最笨的办法:代码级控制:TransactionAspectSup
- ProxyFactory是创建代理类的工厂接口,其中的setProperties方法用来对工厂进行属性设置,但是mybatis内置的两个实现
- 一、前沿如果你学习过其他的编程语言,你就会发现 Java 的语法很是哆嗦,可是我们为什么没有放弃 Java 这门编程语言呢?因为 JVM 是
- 线程可以划分优先级,优先级高的线程得到的CPU资源比较多,也就是CPU优先执行优先级高的线程对象中的任务。设置线程优先级有助于帮助线程规划器
- 一、this关键字的作用this关键字除了可以强调本类中的方法还具有以下作用。1.表示类中的属性2.可以使用关键字调用本类中的构造方法3.t
- using System;using System.Collections.Generic;using System.ComponentMo
- 本文实例讲述了Android非XML形式动态生成、调用页面的方法。分享给大家供大家参考。具体分析如下:这个问题是这样的:我们不使用XML构建
- 由于项目没有设计返回键,一旦进入别的应用,就无法回到桌面。只能通过串口输入input keyevent 4(返回键)来返回桌面,为了方便调试
- 1.Object类是什么?🟪Object 是 Java 类库中的一个特殊类,也是所有类的父类。也就是说,Java 允许把任何类型的
- 对于获取了一大堆字符串但是又不想要里面的html标签怎么办?特别是像博客园这个富文本框中,可以带样式的,取出来的文章内容也是带样式的。但是在
- 异常是 Java 程序中经常遇到的问题,我想每一个 Java 程序员都讨厌异常,一 个异常就是一个 BUG,就要花很多时间来定位异常问题。1
- 本文实例讲述了Android开发之App widget用法。分享给大家供大家参考,具体如下:放在桌面上的控件叫做——App widget,例
- 使用zxing批量在做好的立牌背景图的指定位置上,把指定的文本内容(链接地址、文本等)生成二维码并放在该位置,最后加上立牌编号。步骤:1).
- 本文实例为大家分享了JSON处理工具类的具体代码,供大家参考,具体内容如下import java.io.IOException; impor
- Eureka什么是服务治理为什么需要服务治理?  服务治理是主要针对分布式服务框架的微服务,处理服务调用
- 错误内容:com.alibaba.druid.pool.GetConnectionTimeoutException: wait millis
- 其实这个http下载器的功能已经相当完善了,支持:限速、post投递和上传、自定义http header、设置user agent、设置ra
- 为了实现不同环境构建的不同需求,这里使用到了 profile。因为 profile 能够在构建时修改 pom 的一个子集,或者添加额外的配置
- 在Material Design里,CoordinatorLayout通常用来作为顶层视图,来协调处理各个子View之间的动作,从而实现各种
- 克隆,想必大家都有耳闻,世界上第一只克隆羊多莉就是利用细胞核移植技术将哺乳动物的成年体细胞培育出新个体,甚为神奇。其实在Java中也存在克隆