PropertySource和Enviroment http://blog.csdn.net/u011179993/article/details/51511364
统一抽象资源---Resource http://blog.csdn.net/u011179993/article/details/51531140
SpringApplicationRunListener及其周期 http://blog.csdn.net/u011179993/article/details/51555690
BeanDefinition及读取、注册 http://blog.csdn.net/u011179993/article/details/51598567
http://blog.csdn.net/u011179993/article/category/5623745
BeanDefinitionRegistry 该类的作用主要是向注册表中注册 BeanDefinition 实例,完成 注册的过程。
public interface BeanDefinitionRegistry extends AliasRegistry {
// 关键 -> 往注册表中注册一个新的 BeanDefinition 实例
void registerBeanDefinition(String beanName, BeanDefinition beanDefinition)throws BeanDefinitionStoreException;
// 移除注册表中已注册的 BeanDefinition 实例
void removeBeanDefinition(String beanName) throws NoSuchBeanDefinitionException;
// 从注册中取得指定的 BeanDefinition 实例
BeanDefinition getBeanDefinition(String beanName) throws NoSuchBeanDefinitionException;
// 判断 BeanDefinition 实例是否在注册表中(是否注册)
boolean containsBeanDefinition(String beanName);
// 取得注册表中所有 BeanDefinition 实例的 beanName(标识)
String[] getBeanDefinitionNames();
// 返回注册表中 BeanDefinition 实例的数量
int getBeanDefinitionCount();
// beanName(标识)是否被占用
boolean isBeanNameInUse(String beanName);
}
http://blog.csdn.net/u011179993/article/details/51655057
BeanFactoryPostProcessor 当spring初始化好BenaDefinnitionMap之后,提供了一个接口BeanFactoryPostProcessor,允许我们开发者自定义的去修改BeanFactory中的内容,这也是符合“spring”的开闭原则
public interface BeanFactoryPostProcessor {
/**
* 这里提供了修改beanFacotry的机会
*/
void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException;
}
http://jinnianshilongnian.iteye.com/blog/2000183
PropertySource:属性源,key-value属性对抽象,比如用于配置数据
PropertyResolver 属性解析器,用来根据名字解析其值等
public interface PropertyResolver {
//是否包含某个属性
boolean containsProperty(String key);
//获取属性值 如果找不到返回null
String getProperty(String key);
//获取属性值,如果找不到返回默认值
String getProperty(String key, String defaultValue);
//获取指定类型的属性值,找不到返回null
T getProperty(String key, Class targetType);
//获取指定类型的属性值,找不到返回默认值
T getProperty(String key, Class targetType, T defaultValue);
//获取属性值为某个Class类型,找不到返回null,如果类型不兼容将抛出ConversionException
Class getPropertyAsClass(String key, Class targetType);
//获取属性值,找不到抛出异常IllegalStateException
String getRequiredProperty(String key) throws IllegalStateException;
//获取指定类型的属性值,找不到抛出异常IllegalStateException
T getRequiredProperty(String key, Class targetType) throws IllegalStateException;
//替换文本中的占位符(${key})到属性值,找不到不解析
String resolvePlaceholders(String text);
//替换文本中的占位符(${key})到属性值,找不到抛出异常IllegalArgumentException
String resolveRequiredPlaceholders(String text) throws IllegalArgumentException;
}
http://blog.csdn.net/u011179993/article/details/51598567
BeanDefinition 这个接口描述bean的结构,对应XML中的< bean >或者配置类中的@Bean 它集成了BeanMetadataElement和AttributeAccessor
PropertyValues 包含了一个或者多个PropertyValue对象,通常用作特定的一个目的bean的属性更新
AttributeAccessor接口定义了最基本的对任意对象的元数据的修改或者获取
private void initialize(Object[] sources) {
if (sources != null && sources.length > 0) {
this.sources.addAll(Arrays.asList(sources));
}
//判断是否web环境
//javax.servlet.Servlet"和"org.springframework.web.context.ConfigurableWebApplicationContext"类能被加载到代表为web环境
this.webEnvironment = deduceWebEnvironment();
/*
*通过Thread.currentThread().getContextClassLoader(); 获取到 类加载器
*获取 spring-boot-{v}.jar/META-INF/spring.factories 文件中 org.springframework.context.ApplicationContextInitializer 的值
* org.springframework.boot.context.ConfigurationWarningsApplicationContextInitializer 配置警告应用程序上下文初始化程序
* org.springframework.boot.context.ContextIdApplicationContextInitializer 上下文Id 应用上下文初始化器
* org.springframework.boot.context.config.DelegatingApplicationContextInitializer 委托应用程序上下文初始化程序
* org.springframework.boot.context.embedded.ServerPortInfoApplicationContextInitializer 设置端口
*并通过各自的构造方法实例化
*
*/
setInitializers((Collection) getSpringFactoriesInstances(ApplicationContextInitializer.class));
/*
*获取 spring-boot-{v}.jar/META-INF/spring.factories 文件中 org.springframework.context.ApplicationListener 的值,并实例化
* org.springframework.boot.ClearCachesApplicationListener
* org.springframework.boot.builder.ParentContextCloserApplicationListener
* org.springframework.boot.context.FileEncodingApplicationListener
* org.springframework.boot.context.config.AnsiOutputApplicationListener
* org.springframework.boot.context.config.ConfigFileApplicationListener
* org.springframework.boot.context.config.DelegatingApplicationListener
* org.springframework.boot.liquibase.LiquibaseServiceLocatorApplicationListener
* org.springframework.boot.logging.ClasspathLoggingApplicationListener
* org.springframework.boot.logging.LoggingApplicationListener
*
setListeners((Collection) getSpringFactoriesInstances(ApplicationListener.class));
/*
*获取main方法所在的类
*StackTraceElement[] stackTrace = new RuntimeException().getStackTrace(); 获取 虚拟机栈中 栈桢数组
*/
this.mainApplicationClass = deduceMainApplicationClass();
}
public ConfigurableApplicationContext run(String... args) {
//计时器
StopWatch stopWatch = new StopWatch();
//开始计时
stopWatch.start();
ConfigurableApplicationContext context = null;
FailureAnalyzers analyzers = null;
//用来设置java.awt.headless 属性是true 还是false,是J2SE的一种模式用于在缺少显示屏、键盘
//或者鼠标时的系统配置
configureHeadlessProperty();
/*
*获得运行侦听器 获取 spring-boot-{v}.jar/META-INF/spring.factories 文件中 org.springframework.boot.SpringApplicationRunListener 的值
* org.springframework.boot.context.event.EventPublishingRunListener 事件发布运行侦听器
*在EventPublishingRunListener实例化时将 ApplicationListener的所有实现类添加 到SimpleApplicationEventMulticaster 事件中
*/
SpringApplicationRunListeners listeners = getRunListeners(args);
/*
*启动SpringApplicationRunListeners侦听器
*启动EventPublishingRunListener侦听器
*org.springframework.boot.context.config.ConfigFileApplicationListener onApplicationEvent没有做任何事
*org.springframework.boot.logging.LoggingApplicationListener onApplicationEvent加载 LoggingSystem对象
*org.springframework.boot.context.config.DelegatingApplicationListener onApplicationEvent没有做任何事
*org.springframework.boot.liquibase.LiquibaseServiceLocatorApplicationListener onApplicationEvent没有做任何事
*调用以上监听器的onApplicationEvent 方法
*/
listeners.starting();
try {
//设置应用参数
ApplicationArguments applicationArguments = new DefaultApplicationArguments(
args);
//加载配置
ConfigurableEnvironment environment = prepareEnvironment(listeners,
applicationArguments);
//打印Banner 默认 SpringBootBanner.BANNER
Banner printedBanner = printBanner(environment);
//获取 上下文ApplicationContext
context = createApplicationContext();
//错误处理
analyzers = new FailureAnalyzers(context);
//准备上下文
prepareContext(context, environment, listeners, applicationArguments,
printedBanner);
//刷新上下文
refreshContext(context);
//完成加载
afterRefresh(context, applicationArguments);
//完成侦听器
listeners.finished(context, null);
//停止记时
stopWatch.stop();
//记录启动信息
if (this.logStartupInfo) {
new StartupInfoLogger(this.mainApplicationClass)
.logStarted(getApplicationLog(), stopWatch);
}
return context;
}
catch (Throwable ex) {
handleRunFailure(context, listeners, analyzers, ex);
throw new IllegalStateException(ex);
}
}
private ConfigurableEnvironment prepareEnvironment(
SpringApplicationRunListeners listeners,
ApplicationArguments applicationArguments) {
// 创建和配置环境 StandardServletEnvironment.customizePropertySources获取配置加载顺序
ConfigurableEnvironment environment = getOrCreateEnvironment();
//获取启动配置
configureEnvironment(environment, applicationArguments.getSourceArgs());
/**
*通过 EventPublishingRunListener监听 发送ApplicationEnvironmentPreparedEvent 事件到下面监听器
*加载配置以下配置的值并加载配置,并调用以下监听的 onApplicationEvent 方法
*org.springframework.boot.context.config.ConfigFileApplicationListener 加载配置
* 获取org.springframework.boot.env.EnvironmentPostProcessor配置的值
* org.springframework.boot.cloud.CloudFoundryVcapEnvironmentPostProcessor
* 获取 spring cloud config VCAP_APPLICATION 和 VCAP_SERVICES 的配置
*
* org.springframework.boot.env.SpringApplicationJsonEnvironmentPostProcessor
* 获取${spring.application.json:${SPRING_APPLICATION_JSON:}} 的配置
*
* org.springframework.boot.context.config.ConfigFileApplicationListener
* 添加属性源 [servletConfigInitParams,servletContextInitParams,systemProperties,systemEnvironment,random]
* 获取 org.springframework.boot.env.PropertySourceLoader的值
* org.springframework.boot.env.PropertiesPropertySourceLoader
* org.springframework.boot.env.YamlPropertySourceLoader
* 配置文件路径加载顺序 1.获取用户自定义路径地址(spring.config.location的值) 2.[file:./config/, file:./, classpath:/config/, classpath:/]
* 文件加载顺序 1.获取用户自定义文件名(spring.config.name的值) 2.默认文件名(application)
* 文件后缀加载顺序: [properties, xml, yml, yaml]
* 1.file:./config/application.properties
* 2.file:./config/application.xml
* 3.file:./config/application.yml
* 4.file:./config/application.yaml
* file:./application.properties
* file:./application.xml
* file:./application.yml
* file:./application.yaml
* file:./application.yaml
* classpath:/config/application.properties
* classpath:/config/application.xml
* classpath:/config/application.yml
* classpath:/config/application.yaml
* //我配置的是放在 classpath下的
* classpath:/application.properties
* classpath:/application-dev.properties
* 读取完配置,过滤spring bean
* 将配置绑定到 SpringApplication
*org.springframework.boot.context.config.AnsiOutputApplicationListener
*org.springframework.boot.logging.LoggingApplicationListener
*org.springframework.boot.logging.ClasspathLoggingApplicationListener
*org.springframework.boot.autoconfigure.BackgroundPreinitializer
*org.springframework.boot.context.config.DelegatingApplicationListener
*org.springframework.boot.context.FileEncodingApplicationListener
*
*
*
*/
listeners.environmentPrepared(environment);
if (!this.webEnvironment) {
environment = new EnvironmentConverter(getClassLoader())
.convertToStandardEnvironmentIfNecessary(environment);
}
return environment;
}
/**
* 配置优先级
* StandardServletEnvironment.customizePropertySources
* SERVLET_CONFIG_PROPERTY_SOURCE_NAME > SERVLET_CONTEXT_PROPERTY_SOURCE_NAME > JNDI_PROPERTY_SOURCE_NAME > 系统属性 > 环境变量
* servlet配置属性源名称 > servlet上下文属性源名称 > jndi属性源名称 > 系统属性 > 环境变量
**/
@Override
protected void customizePropertySources(MutablePropertySources propertySources) {
propertySources.addLast(new StubPropertySource(SERVLET_CONFIG_PROPERTY_SOURCE_NAME));
propertySources.addLast(new StubPropertySource(SERVLET_CONTEXT_PROPERTY_SOURCE_NAME));
if (JndiLocatorDelegate.isDefaultJndiEnvironmentAvailable()) {
propertySources.addLast(new JndiPropertySource(JNDI_PROPERTY_SOURCE_NAME));
}
//获取系统属性、环境变量
super.customizePropertySources(propertySources);
}
protected void configureEnvironment(ConfigurableEnvironment environment,
String[] args) {
//加载命令行属性
configurePropertySources(environment, args);
//获取 spring.profiles.active 的配置
configureProfiles(environment, args);
}
@Override
public void postProcessEnvironment(ConfigurableEnvironment environment,
SpringApplication application) {
addPropertySources(environment, application.getResourceLoader());
//加载配置
configureIgnoreBeanInfo(environment);
绑定资源
bindToSpringApplication(environment, application);
}
protected ConfigurableApplicationContext createApplicationContext() {
Class> contextClass = this.applicationContextClass;
if (contextClass == null) {
try {
/**
* web项目 初始化 org.springframework.boot.context.embedded.AnnotationConfigEmbeddedWebApplicationContext
* java项目 初始化 org.springframework.context.annotation.AnnotationConfigApplicationContext
**/
contextClass = Class.forName(this.webEnvironment
? DEFAULT_WEB_CONTEXT_CLASS : DEFAULT_CONTEXT_CLASS);
}
catch (ClassNotFoundException ex) {
throw new IllegalStateException(
"Unable create a default ApplicationContext, "
+ "please specify an ApplicationContextClass",
ex);
}
}
//初始化AnnotationConfigEmbeddedWebApplicationContext 注释Bean定义阅读器 类路径Bean定义扫描器
return (ConfigurableApplicationContext) BeanUtils.instantiate(contextClass);
}
private void prepareContext(ConfigurableApplicationContext context,
ConfigurableEnvironment environment, SpringApplicationRunListeners listeners,
ApplicationArguments applicationArguments, Banner printedBanner) {
/**
* 注释Bean定义阅读器
* 类路径Bean定义扫描器
**/
context.setEnvironment(environment);
//上下文后处理
postProcessApplicationContext(context);
//初始化上下文
applyInitializers(context);
//上下文准备完成
listeners.contextPrepared(context);
if (this.logStartupInfo) {
logStartupInfo(context.getParent() == null);
logStartupProfileInfo(context);
}
//添加特殊的bean
context.getBeanFactory().registerSingleton("springApplicationArguments",
applicationArguments);
if (printedBanner != null) {
context.getBeanFactory().registerSingleton("springBootBanner", printedBanner);
}
// Load the sources
Set sources = getSources();
Assert.notEmpty(sources, "Sources must not be empty");
//将bean加载到应用程序上下文中。
load(context, sources.toArray(new Object[sources.size()]));
//将context添加到监听中
listeners.contextLoaded(context);
}
@Override
public void refresh() throws BeansException, IllegalStateException {
synchronized (this.startupShutdownMonitor) {
//准备刷新上下文
prepareRefresh();
//告诉子类刷新内部bean工厂
ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();
// 准备在这个上下文中使用的bean工厂
prepareBeanFactory(beanFactory);
try {
// 允许在上下文子类中对bean工厂进行后处理。
postProcessBeanFactory(beanFactory);
// 在上下文中调用在Bean中注册的工厂处理器
invokeBeanFactoryPostProcessors(beanFactory);
// Register bean processors that intercept bean creation.
registerBeanPostProcessors(beanFactory);
// Initialize message source for this context.
initMessageSource();
// Initialize event multicaster for this context.
initApplicationEventMulticaster();
// Initialize other special beans in specific context subclasses.
onRefresh();
// Check for listener beans and register them.
registerListeners();
// Instantiate all remaining (non-lazy-init) singletons.
finishBeanFactoryInitialization(beanFactory);
// Last step: publish corresponding event.
finishRefresh();
}
catch (BeansException ex) {
if (logger.isWarnEnabled()) {
logger.warn("Exception encountered during context initialization - " +
"cancelling refresh attempt: " + ex);
}
// Destroy already created singletons to avoid dangling resources.
destroyBeans();
// Reset 'active' flag.
cancelRefresh(ex);
// Propagate exception to caller.
throw ex;
}
finally {
// Reset common introspection caches in Spring's core, since we
// might not ever need metadata for singleton beans anymore...
resetCommonCaches();
}
}
}
protected void prepareRefresh() {
this.startupDate = System.currentTimeMillis();
this.closed.set(false);
this.active.set(true);
if (logger.isInfoEnabled()) {
logger.info("Refreshing " + this);
}
// 在上下文环境中初始化任何占位符属性源
initPropertySources();
// 验证所有标记为必需的属性是可解析的
getEnvironment().validateRequiredProperties();
this.earlyApplicationEvents = new LinkedHashSet();
}