spring context架构--静态结构

简介: 概念Context也就是我们常说的spring容器,打个比方,context就像是一家公司,beans则是公司的工厂,除了工厂,公司还有翻译,仓库以及办公场所等等。下面就看看context的主要构成部件。Context构成部件上图是ApplicationContext的实体静态结构,它继承了六个实体。虽然是继承,但其实context和他们的关系更像是聚

概念

Context也就是我们常说的spring容器,打个比方,context就像是一家公司,beans则是公司的工厂,除了工厂,公司还有翻译,仓库以及办公场所等等。

下面就看看context的主要构成部件。

Context构成部件

spring context

上图是ApplicationContext的实体静态结构,它继承了六个实体。虽然是继承,但其实context和他们的关系更像是聚合。Spring使用继承主要是为了在context上也同时体现这6个实体的特征。在实现层面,context事实上是个包装类,最终通过聚合的实体类完成相应行为,而ApplicationContext接口本身并没有什么实质意义的方法。

public interface ApplicationContext extends EnvironmentCapable, ListableBeanFactory, HierarchicalBeanFactory,
        MessageSource, ApplicationEventPublisher, ResourcePatternResolver {

    String getId();

    String getApplicationName();

    String getDisplayName();

    long getStartupDate();

    ApplicationContext getParent();

    AutowireCapableBeanFactory getAutowireCapableBeanFactory() throws IllegalStateException;
}

ApplicationContext的实质功能都是继承于以下6个实体:

  1. MessageSource: 用于国际化的接口,可以将其理解为公司的翻译。用户可以通过bean配置自定义MessageSource–要求name为“messageSource”,spring会在容器refresh时自动探测并且初始化它。

  2. ApplicationEventPublisher: 用于发布应用事件,例如ContextRefreshed,stopped, started等。它就像是企业邮箱,通过它可以接收到公司的事件通知。用户通过配置ApplicationListener类型的bean即可订阅这类事件,spring通过getBeansOfType取得所有的listener,并依次通知。

  3. ResourcePatternResolver: 对ResourceLoader的扩展,后者只支持对具体路径资源的加载,而前者则支持对某pattern路径资源的加载,默认是ant风格的模式。Resource是特指配置文件,或者class路径(里的扫描路径)。我们可以将它理解为打单的机器,将各个地方发过来的单子打出来。

  4. ListableBeanFactory和HierachicalBeanFactory: 自然的context也是个工厂,context里持有的依然是DefaultListableBeanFactory,通过它完成工厂的相应行为。介绍下这两个工厂,前者主要用于取得批量bean,比如getBeansOfType;后一个工厂则主要体现层级概念,但是context的parentFactory也是一个context,这是因为context具有beanFactory的所有特征。

  5. EnviromentCapable: 则类似是公司的行政部门,负责办公场所等设施维护。它关联着Enviroment。Enviroment也类似context是个包装类,虽然继承了PropertyResolver,但在实现类里是委托给ConfigurablePropertyResolver处理的。Enviroment代表应用环境,比如测试环境还是生产环境又或者开发环境。
    用户可以通过或者@Profile指定某个配置的profile。然后通过activeProfile指定应用环境,从而会enable相应profile的beans
    activeProfile可以通过5种方式指定:
    1). servlet config init param,这种方式只适用于spring mvc的dispatcherServlet配置上
    2). servlet context init param
    3). system property
    4). system env。以上四种方式设置的key均为spring.profiles.active
    5.) @ActiveProfile,这种方式只适用于junit单元测试

Context通过继承获得了工厂,事件发布,环境定义,资源加载以及国际化的能力。

context静态结构

这一节我只抽一些比较重要的接口的源码讲述,主要还是注重概念和原理,后面会专门出一篇讲context的动态处理过程,那里面会对具体实现类做详述。这一篇泽主要是建立对context静态类结构的理解。

下图是web application context的类图,可以和构成部件的结构图结合着看,上面的图每个实体都是spring context的一个接口。
context类结构
常见的WebApplicationContext实现主要有两个,分别是XmlWebApplicationContext和AnnotationConfigWebApplicationContext。他们共同的父类为AbstractRefreshableWebApplicationContext,我们从它出发,看一下context的类结构。

LifeCycle和Closeable代表着容器的整个生命周期,被ConfigurableApplicationContext继承,使继承该接口的context具有生命周期的特征。

public interface Lifecycle {
    void start();

    void stop();

    boolean isRunning();
}

用户可以配置Lifecycle类型bean,在context start以及stop时也会相应的启动LifeCycle bean的start或者stop。对于允许autoStartup的SmartLifecycle,context refresh的过程中会自动启动。

ConfigurableApplicationContext继承了生命周期,代表一个可以修改相关属性行为的context,上一节提到的Enviroment和ApplicationListener等等都可以通过它设置。同时整个context的核心refresh方法也是定义在他里面,所以所有的context都继承了它。

public interface ConfigurableApplicationContext extends ApplicationContext, Lifecycle, Closeable {
       void setId(String id);

    void setParent(ApplicationContext parent);

    ConfigurableEnvironment getEnvironment();

    void setEnvironment(ConfigurableEnvironment environment);

//添加bean后处理器,[beans架构](http://blog.csdn.net/szwandcj/article/details/50688616)架构里详细讲过后处理器
       void addBeanFactoryPostProcessor(BeanFactoryPostProcessor beanFactoryPostProcessor);

    void addApplicationListener(ApplicationListener<?> listener);

//context核心方法,refresh是整个容器构建的过程
    void refresh() throws BeansException, IllegalStateException;

//注册jdk进程退出时的hook
    void registerShutdownHook();

    void close();

    boolean isActive();

    ConfigurableListableBeanFactory getBeanFactory() throws IllegalStateException;
}

实现这个接口的AbstractApplicationContext很自然的也承担了context的绝大部分职能,上一节提过的context的六个方面的功能就几乎都是由它实现,除BeanFactory以外,其它的4个实体都是聚合在这个类里。而DefaultListableBeanFactory–那个最著名的BeanFactory则是由它的子类AbstractRefreshableApplicationContext生成,它自己通过模板方法使用。

AbstractRefreshableApplicationContext定义了loadBeanDefinitions模板方法,由具体的实现提供。它创建bean factory并load配置文件。beans配置文件load过程参考这里

AbstractRefreshableConfigApplicationContext主要用于设置资源的路径。几乎所有的ApplicationContext都会继承这个类,除非不需要读取配置资源。

AbstractRefreshableWebApplicationContext主要针对web的一些特性提供一些context属性行为设置能力,例如覆盖默认的enviroment(默认的由AbstractApplicationContext提供),生成StandardServletEnviroment,又例如覆盖ResourcePatternResolver以及对BeanFactory的后处理等。

ConfigurableWebApplicationContext则是web application context的通用接口,为了支持spring-mvc和spring-web定义了一些set方法,例如设置configLocations(ConfigurableApplicationContext接口并不支持setConfigLocation,因为这个接口是跟着ClassPathXmlApplicationContext一起发布的,而web和mvc是后来加上的功能)和nameSpace等。可以把它看成是专门支持web和mvc的一个接口(web和mvc中生成context是通过class.newInstance的无参构造,所以无法将这些信息作为参数传入,必须显示提供set方法以设置属性)

public interface ConfigurableWebApplicationContext extends WebApplicationContext, ConfigurableApplicationContext {

    void setServletContext(ServletContext servletContext);

    void setServletConfig(ServletConfig servletConfig);

    ServletConfig getServletConfig();

    void setNamespace(String namespace);

    String getNamespace();

    void setConfigLocation(String configLocation);

    void setConfigLocations(String... configLocations);

    String[] getConfigLocations();
}

Enviroment实现

再具体说下Enviroment的实现,还是挺有意思的。下图是enviroment的序列图
enviroment

主要说下MutablePropertySources和PropertySource。

最终属性值是通过PropertySource取得,它有多种实现,分别代表一种来源的属性。第二节提到的4种来源则分别对应着ServletConfigPropertySource,ServletContextPropertySource,MapPropertySource和SystemEnvironmentPropertySource。

MutablePropertySources里持有一个property的list,对它迭代直到从一个PropertySource里取出对应key的值就停止,从这可以就看出spring.profiles.active是有先后优先级的。

有兴趣的可以看下源码,会稍微有些繁琐。有一点需要注意:spring web和mvc模块会在refresh applicationContext之前调用相应enviroment#initPropertySources生成ServletConfigPropertySource和ServletContextPropertySource。相应的MapPropertySource和SystemEnvironmentPropertySource则是默认生成的。所以如果不是web项目,就只能通过配置system properties或者env指定activeProfile。

目录
相关文章
|
1月前
|
XML 安全 Java
|
2月前
|
数据采集 监控 前端开发
二级公立医院绩效考核系统源码,B/S架构,前后端分别基于Spring Boot和Avue框架
医院绩效管理系统通过与HIS系统的无缝对接,实现数据网络化采集、评价结果透明化管理及奖金分配自动化生成。系统涵盖科室和个人绩效考核、医疗质量考核、数据采集、绩效工资核算、收支核算、工作量统计、单项奖惩等功能,提升绩效评估的全面性、准确性和公正性。技术栈采用B/S架构,前后端分别基于Spring Boot和Avue框架。
110 5
|
1月前
|
Java 开发者 微服务
从单体到微服务:如何借助 Spring Cloud 实现架构转型
**Spring Cloud** 是一套基于 Spring 框架的**微服务架构解决方案**,它提供了一系列的工具和组件,帮助开发者快速构建分布式系统,尤其是微服务架构。
208 69
从单体到微服务:如何借助 Spring Cloud 实现架构转型
|
9天前
|
搜索推荐 NoSQL Java
微服务架构设计与实践:用Spring Cloud实现抖音的推荐系统
本文基于Spring Cloud实现了一个简化的抖音推荐系统,涵盖用户行为管理、视频资源管理、个性化推荐和实时数据处理四大核心功能。通过Eureka进行服务注册与发现,使用Feign实现服务间调用,并借助Redis缓存用户画像,Kafka传递用户行为数据。文章详细介绍了项目搭建、服务创建及配置过程,包括用户服务、视频服务、推荐服务和数据处理服务的开发步骤。最后,通过业务测试验证了系统的功能,并引入Resilience4j实现服务降级,确保系统在部分服务故障时仍能正常运行。此示例旨在帮助读者理解微服务架构的设计思路与实践方法。
55 16
|
12天前
|
监控 JavaScript 数据可视化
建筑施工一体化信息管理平台源码,支持微服务架构,采用Java、Spring Cloud、Vue等技术开发。
智慧工地云平台是专为建筑施工领域打造的一体化信息管理平台,利用大数据、云计算、物联网等技术,实现施工区域各系统数据汇总与可视化管理。平台涵盖人员、设备、物料、环境等关键因素的实时监控与数据分析,提供远程指挥、决策支持等功能,提升工作效率,促进产业信息化发展。系统由PC端、APP移动端及项目、监管、数据屏三大平台组成,支持微服务架构,采用Java、Spring Cloud、Vue等技术开发。
|
1月前
|
负载均衡 Java 开发者
深入探索Spring Cloud与Spring Boot:构建微服务架构的实践经验
深入探索Spring Cloud与Spring Boot:构建微服务架构的实践经验
164 5
|
3月前
|
Java Spring
Spring底层架构源码解析(三)
Spring底层架构源码解析(三)
214 5
|
3月前
|
XML Java 数据格式
Spring底层架构源码解析(二)
Spring底层架构源码解析(二)
|
3月前
|
JSON 前端开发 Java
Spring Boot框架中的响应与分层解耦架构
在Spring Boot框架中,响应与分层解耦架构是两个核心概念,它们共同促进了应用程序的高效性、可维护性和可扩展性。
82 3
|
3月前
|
存储 前端开发 数据库
一文搞懂SaaS应用架构:应用服务、应用结构、应用交互设计
【10月更文挑战第21天】本文介绍了 SaaS 应用服务的多租户服务、安全服务和更新与维护服务,以及 SaaS 应用的前后端结构和交互设计。多租户服务涉及数据隔离和资源分配;安全服务包括身份认证与授权及数据安全;更新与维护服务涵盖版本管理和技术支持。前端结构关注用户界面设计和前端技术选型;后端结构则涉及微服务架构和数据库管理。交互设计强调租户与应用的交互和应用内部模块间的交互。
296 0