这个接口描述bean的结构,对应XML中的< bean >或者配置类中的@Bean
它继承了BeanMetadataElement和AttributeAccessor接口,如下图
【AttributeAccessor接口】
类似于map,具有保存和访问name/value属性的能力。
public interface AttributeAccessor {
void setAttribute(String name, Object value);
Object getAttribute(String name);
Object removeAttribute(String name);
boolean hasAttribute(String name);
String[] attributeNames();
}
【BeanMetadataElement接口】
具有访问source(配置源)的能力。
public interface BeanMetadataElement {
Object getSource();
}
【BeanDefinition接口】
定义了设置、获取一个BeanDefinition属性方便方法(比如获取scope属性直接使用getScope())
public interface BeanDefinition extends AttributeAccessor, BeanMetadataElement {
//单例或原型
String SCOPE_SINGLETON = ConfigurableBeanFactory.SCOPE_SINGLETON;
String SCOPE_PROTOTYPE = ConfigurableBeanFactory.SCOPE_PROTOTYPE;
//Bean角色
int ROLE_APPLICATION = 0;
int ROLE_SUPPORT = 1;
int ROLE_INFRASTRUCTURE = 2;
// 返回/设置父BeanDefinition
String getParentName();
void setParentName(String parentName);
//返回/设置 当前的BeanClassName(不等于最终Bean的名称)
String getBeanClassName();
void setBeanClassName(String beanClassName);
//返回设置 factory bean name
String getFactoryBeanName();
void setFactoryBeanName(String factoryBeanName);
String getFactoryMethodName();
void setFactoryMethodName(String factoryMethodName);
String getScope();
void setScope(String scope);
boolean isLazyInit();
void setLazyInit(boolean lazyInit);
String[] getDependsOn();
void setDependsOn(String... dependsOn);
boolean isAutowireCandidate();
void setAutowireCandidate(boolean autowireCandidate);
boolean isPrimary();
void setPrimary(boolean primary);
ConstructorArgumentValues getConstructorArgumentValues();
MutablePropertyValues getPropertyValues();
boolean isSingleton();
boolean isPrototype();
boolean isAbstract();
int getRole();
String getDescription();
String getResourceDescription();
BeanDefinition getOriginatingBeanDefinition();
}
这个接口可以获取BeanDefinition注解相关数据
public interface AnnotatedBeanDefinition extends BeanDefinition {
AnnotationMetadata getMetadata();
MethodMetadata getFactoryMethodMetadata();
}
这个抽象类的构造方法设置了BeanDefinition的默认属性,重写了equals,hashCode,toString方法。
可以从父BeanDefinition中集成构造方法,属性等
代表一个从配置源(XML,Java Config等)中生成的BeanDefinition
GenericBeanDefinition是自2.5以后新加入的bean文件配置属性定义类,是ChildBeanDefinition和RootBeanDefinition更好的替代者
对应注解@Bean
@Component("t")
public class Tester {
public static void main(String[] args) throws Exception {
AnnotatedGenericBeanDefinition beanDefinition=new AnnotatedGenericBeanDefinition(Tester.class);
System.out.println(beanDefinition.getMetadata().getAnnotationTypes());
System.out.println(beanDefinition.isSingleton());
System.out.println(beanDefinition.getBeanClassName());
}
}
==========输出==============
[org.springframework.stereotype.Component]
true
Tester
持有一个BeanDefinition,名称,和别名数组。在Spring内部,它用来临时保存BeanDefinition来传递BeanDefinition。
它的部分方法如下:
public class BeanDefinitionHolder implements BeanMetadataElement {
private final BeanDefinition beanDefinition;
private final String beanName;
private final String[] aliases;
public BeanDefinitionHolder(BeanDefinition beanDefinition, String beanName, String[] aliases) {
Assert.notNull(beanDefinition, "BeanDefinition must not be null");
Assert.notNull(beanName, "Bean name must not be null");
this.beanDefinition = beanDefinition;
this.beanName = beanName;
this.aliases = aliases;
}
public BeanDefinition getBeanDefinition() {
return this.beanDefinition;
}
public String getBeanName() {
return this.beanName;
}
public String[] getAliases() {
return this.aliases;
}
//其他方法...省略
这个接口定义了‘注册/获取BeanDefinition’的方法 。
接口定义:
public interface BeanDefinitionRegistry extends AliasRegistry {
//注册一个BeanDefinition
void registerBeanDefinition(String beanName, BeanDefinition beanDefinition)
throws BeanDefinitionStoreException;
//根据name,从自己持有的多个BeanDefinition 中 移除一个
void removeBeanDefinition(String beanName) throws NoSuchBeanDefinitionException;
// 获取某个BeanDefinition
BeanDefinition getBeanDefinition(String beanName) throws NoSuchBeanDefinitionException;
boolean containsBeanDefinition(String beanName);//是否包含
String[] getBeanDefinitionNames();//获取所有名称
int getBeanDefinitionCount();//获取持有的BeanDefinition数量
boolean isBeanNameInUse(String beanName); //判断某个BeanDefinition是否在使用
}
实现类一般使用Map保存多个BeanDefinition,如下:
Map<String, BeanDefinition> beanDefinitionMap = new HashMap<String, BeanDefinition>();
实现类有SimpleBeanDefinitionRegistry,DefaultListableBeanFactory,GenericApplicationContext等。
SimpleBeanDefinitionRegistry是最基本的实现类。
使用:
public static void main(String[] args) throws Exception {
//实例化SimpleBeanDefinitionRegistry
SimpleBeanDefinitionRegistry registry = new SimpleBeanDefinitionRegistry();
//注册两个BeanDefinition
BeanDefinition definition_1 = new GenericBeanDefinition();
registry.registerBeanDefinition("d1", definition_1);
BeanDefinition definition_2 = new RootBeanDefinition();
registry.registerBeanDefinition("d2", definition_2);
//方法测试
System.out.println(registry.containsBeanDefinition("d1"));//true
System.out.println(registry.getBeanDefinitionCount());//2
System.out.println(Arrays.toString(registry.getBeanDefinitionNames()));//[d1, d2]
}
================结果==================
true
2
[d1, d2]
BeanDefinitionRegistry接口一次只能注册一个BeanDefinition,而且只能自己构造BeanDefinition类来注册。BeanDefinitionReader解决了这些问题,它一般可以使用一个BeanDefinitionRegistry构造,然后通过#loadBeanDefinitions(..)等方法,把“配置源”转化为多个BeanDefinition并注册到BeanDefinitionRegistry中 。
可以说BeanDefinitionReader帮助BeanDefinitionRegistry实现了高效、方便的注册BeanDefinition。
public interface BeanDefinitionReader {
//获取BeanDefinitionRegistry
BeanDefinitionRegistry getRegistry();
ResourceLoader getResourceLoader();
ClassLoader getBeanClassLoader();
//获取Bean的名称生成器
BeanNameGenerator getBeanNameGenerator();
//载入BeanDefinition
int loadBeanDefinitions(Resource resource) throws BeanDefinitionStoreException;
//载入BeanDefinition
int loadBeanDefinitions(Resource... resources) throws BeanDefinitionStoreException;
//载入BeanDefinition
int loadBeanDefinitions(String location) throws BeanDefinitionStoreException;
//载入BeanDefinition
int loadBeanDefinitions(String... locations) throws BeanDefinitionStoreException;
}
AbstractBeanDefinitionReader:实现了EnvironmentCapable,提供了获取/设置环境的方法。定义了一些通用方法,使用策略模式,将一些具体方法放到子类实现。
XmlBeanDefinitionReader:读取XML文件定义的BeanDefinition
PropertiesBeanDefinitionReader:可以从属性文件,Resource,Property对象等读取BeanDefinition
GroovyBeanDefinitionReader:可以读取Groovy语言定义的Bean
把xml配置文件转化成beanDefinition
这个类就不做演示了
可以从Properties文件读取BeanDefinition,文件可以是如下结构:
#bean名称.(属性)=值
==========================
mybean.(class)=PropertyReaderTest
mybean.(abstract)=true
mybean.(lazy-init)=true
mybean.(scope)=prototype
程序实例:
public class PropertyReaderTest {
public static void main(String[] args) {
//定义BeanDefinitionRegistry
BeanDefinitionRegistry registry = new SimpleBeanDefinitionRegistry();
//使用BeanDefinitionRegistry实例 构造BeanDefinitionReader
PropertiesBeanDefinitionReader reader = new PropertiesBeanDefinitionReader(registry);
//载入文件
reader.loadBeanDefinitions("config.properties");
System.out.println(registry.getBeanDefinition("mybean"));
}
}
================输出=================
.......//日志信息 略
Generic bean: class [PropertyReaderTest]; scope=prototype; abstract=true; lazyInit=true; autowireMode=0; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=null; factoryMethodName=null; initMethodName=null; destroyMethodName=null
注解相关的BeanDefinitionReader并没有实现BeanDefinitionReader接口,有如下类:
使用如下:
SimpleBeanDefinitionRegistry registry= new SimpleBeanDefinitionRegistry();
AnnotatedBeanDefinitionReader reader = new AnnotatedBeanDefinitionReader(registry);
reader.register(Tester.class);
System.out.println( registry.getBeanDefinitionCount());
=========================
7
我们只注册了一个Bean,为什么出现7个呢?原来是AnnotatedBeanDefinitionReader 向BeanDefinitionRegistry自动注册了6个BeanDefinition。这个在以后章节中详解。
ClassPathBeanDefinitionScanner 继承自ClassPathScanningCandidateComponentProvider
类结构如下:
【ClassPathScanningCandidateComponentProvider】
这个类的findCandidateComponents可以扫描到@Component @Repository @Service @Controller 的BeanDefinition,并加入Set 集合中
ClassPathScanningCandidateComponentProvider provider=new ClassPathScanningCandidateComponentProvider(true);
Set definitionSet= provider.findCandidateComponents("com.jazz.web");
System.out.println(definitionSet.size());
【ClassPathBeanDefinitionScanner】
这里写代码片