Spring源码探究 | 一、启程

启程

1、前言

接下来的一段时间内,我会把我学习spring源码的相关心得进行整理,发布在我的博客里。为自己,能够记录自己的成长路程,能够温故知新。为他人,能够分享下我的相关理解和经验,希望对想深挖spring的同学能起到一点帮助。
spring的整个体系很庞大,学习需要方法。我这里总结的方法就是:

  • 1、寻找入口,即寻找入手学习的点。这个对于spring就是核心流程的入口点:IOC、DI和AOP、MVC是主体如何实现的。
  • 2、画图。找到入口后,把每个关键的类以及其方法进行记录,最终梳理成一个时序图。对于一些关键的类,画类图来明确他们之间的继承或者实现关系,帮助我们整体来看spring的继承体系。
  • 3、多看几遍,多走几遍,温故知新。根据自己总结的时序图,多看几遍源码,根据debug多走几遍大体流程,了解每个核心方法中所作的事情。
  • 4、坚持下去。看源码的过程是非常枯燥的,对于很陌生的代码,经常会有想放弃的想法,但请坚持,坚持到你能有所收获。
  • 5、学会总结。在通过一段时间的学习,请总结下spring中的核心功能的实现原理、总结下spring中使用到的设计模式和设计思想。因为通过学习源码,一是能深入了解spring,二是能够通过前辈们的写下的代码了解其设计思想,并在以后实践到自己的项目中。

希望在学习spring后,能达到知其然知其所以然的境界。(知道它是这样的,更知道它为什么是这样的。)

2、设计初衷

在学习任何一门语言或者框架的时候,我们要知道我们学习的对象它的设计初衷是什么,接下来的一系列的代码都是围绕这个初衷去实现并不断的去迭代。那么Spring的设计初衷是什么。
Spring 是为解决企业级应用开发的复杂性而设计,它可以做很多事,但归根到底支撑 Spring 的仅
仅是少许的基本理念,而所有的这些基本理念都能可以追溯到一个最根本的使命:简化开发(其实任何一个框架的存在都是为了简化开发)。
这是一个郑重的承诺,其实许多框架都声称在某些方面做了简化。而 Spring 则立志于全方面的简化 Java 开发。
对此,她主要采取了 4 个关键策略

  • 1、基于 POJO 的轻量级和最小侵入性编程;
  • 2、通过依赖注入和面向接口松耦合;
  • 3、基于切面和惯性进行声明式编程;
  • 4、通过切面和模板减少样板式代码;

而他主要是通过:面向 Bean(BOP)、依赖注入(DI)以及面向切面(AOP)这三种方式来达成的。
这里大概介绍下这几个思想。

3、BOP 编程伊始

Spring 是面向 Bean 的编程(Bean Oriented Programming, BOP),Bean 在 Spring 中才是真正的
主角。Bean 在 Spring 中作用就像 Object 对 OOP 的意义一样,Spring 中没有 Bean 也就没有 Spring
存在的意义。Spring 提供了 IOC 容器通过配置文件或者注解的方式来管理对象之间的依赖关系。

控制反转(其中最常见的实现方式叫做依赖注入(Dependency Injection,DI),还有一种方式叫
“依赖查找”(Dependency Lookup,DL),她在 C++、Java、PHP 以及.NET 中都运用。在最早的
Spring 中是包含有依赖注入方法和依赖查询的,但因为依赖查询使用频率过低,不久就被 Spring 移除
了,所以在 Spring 中控制反转也被直接称作依赖注入),她的基本概念是:不创建对象,但是描述创建
它们的方式。在代码中不直接与对象和服务连接,但在配置文件中描述哪一个组件需要哪一项服务。容
器 (在 Spring 框架中是 IOC 容器)负责将这些联系在一起。

在典型的 IOC 场景中,容器创建了所有对象,并设置必要的属性将它们连接在一起,决定什么时间调用方法。

4、依赖注入的基本概念

Spring 设计的核心 org.springframework.beans 包(架构核心是 org.springframework.core
包),它的设计目标是与 JavaBean 组件一起使用。这个包通常不是由用户直接使用,而是由服务器将
其用作其他多数功能的底层中介。下一个最高级抽象是 BeanFactory 接口,它是工厂设计模式的实现,
允许通过名称创建和检索对象。BeanFactory 也可以管理对象之间的关系。

BeanFactory 最底层支持两个对象模型。

  • 1,单例:提供了具有特定名称的全局共享实例对象,可以在查询时对其进行检索。Singleton 是默
    认的也是最常用的对象模型。
  • 2,原型:确保每次检索都会创建单独的实例对象。在每个用户都需要自己的对象时,采用原型模式。
    Bean 工厂的概念是 Spring 作为 IOC 容器的基础。IOC 则将处理事情的责任从应用程序代码转移到
    框架。

5、AOP 编程理念

面向切面编程,即 AOP,是一种编程思想,它允许程序员对横切关注点或横切典型的职责分界线的
行为(例如日志和事务管理)进行模块化。AOP 的核心构造是方面(切面),它将那些影响多个类的行
为封装到可重用的模块中。

AOP 和 IOC 是补充性的技术,它们都运用模块化方式解决企业应用程序开发中的复杂问题。在典
型的面向对象开发方式中,可能要将日志记录语句放在所有方法和 Java 类中才能实现日志功能。在 AOP
方式中,可以反过来将日志服务模块化,并以声明的方式将它们应用到需要日志的组件上。当然,优势
就是 Java 类不需要知道日志服务的存在,也不需要考虑相关的代码。所以,用 Spring AOP 编写的应
用程序代码是松散耦合的。

AOP 的功能完全集成到了 Spring 事务管理、日志和其他各种特性的上下文中。
AOP 编程的常用场景有:Authentication(权限认证)、Auto Caching(自动缓存处理)、Error Handling
(统一错误处理)、Debugging(调试信息输出)、Logging(日志记录)、Transactions(事务处理)
等。

6、Spring5 系统架构

Spring 总共大约有 20 个模块,由 1300 多个不同的文件构成。而这些组件被分别整合在核心容器(Core
Container)、AOP(Aspect Oriented Programming)和设备支持(Instrmentation)、数据访问
及集成(Data Access/Integeration)、Web、报文发送(Messaging)、Test,6 个模块集合中。以
下是 Spring 5 的模块结构图:
在这里插入图片描述
组成 Spring 框架的每个模块集合或者模块都可以单独存在,也可以一个或多个模块联合实现。每个模
块的组成和功能如下:

6.1、核心容器

由spring-beans、spring-core、spring-context和spring-expression(Spring Expression Language,
SpEL) 4 个模块组成。

  • spring-core 和 spring-beans 模块是 Spring 框架的核心模块,包含了控制反转(Inversion of
    Control, IOC)和依赖注入(Dependency Injection, DI)。BeanFactory 接口是 Spring 框架中的核
    心接口,它是工厂模式的具体实现。BeanFactory 使用控制反转对应用程序的配置和依赖性规范与实际
    的应用程序代码进行了分离。但 BeanFactory 容器实例化后并不会自动实例化 Bean,只有当 Bean 被
    使用时 BeanFactory 容器才会对该 Bean 进行实例化与依赖关系的装配。
  • spring-context 模块构架于核心模块之上,他扩展了 BeanFactory,为她添加了 Bean 生命周期控
    制、框架事件体系以及资源加载透明化等功能。此外该模块还提供了许多企业级支持,如邮件访问、远
    程访问、任务调度等,ApplicationContext 是该模块的核心接口,她的超类是 BeanFactory。与
    BeanFactory 不同,ApplicationContext 容器实例化后会自动对所有的单实例 Bean 进行实例化与依
    赖关系的装配,使之处于待用状态。
  • spring-context-support 模块是对 Spring IOC 容器的扩展支持,以及 IOC 子容器。
  • spring-context-indexer 模块是 Spring 的类管理组件和 Classpath 扫描。
  • spring-expression 模块是统一表达式语言(EL)的扩展模块,可以查询、管理运行中的对象,同
    时也方便的可以调用对象方法、操作数组、集合等。它的语法类似于传统 EL,但提供了额外的功能,最
    出色的要数函数调用和简单字符串的模板函数。这种语言的特性是基于 Spring 产品的需求而设计,他
    可以非常方便地同 Spring IOC 进行交互。

6.2、AOP 和设备支持

由 spring-aop、spring-aspects 和 spring-instrument 3 个模块组成

  • spring-aop 是 Spring 的另一个核心模块,是 AOP 主要的实现模块。作为继 OOP 后,对程序员影
    响最大的编程思想之一,AOP 极大地开拓了人们对于编程的思路。在 Spring 中,他是以 JVM 的动态
    代理技术为基础,然后设计出了一系列的 AOP 横切实现,比如前置通知、返回通知、异常通知等,同
    时,Pointcut 接口来匹配切入点,可以使用现有的切入点来设计横切面,也可以扩展相关方法根据需求
    进行切入。
  • spring-aspects 模块集成自 AspectJ 框架,主要是为 Spring AOP 提供多种 AOP 实现方法。
  • spring-instrument 模块是基于 JAVA SE 中的"java.lang.instrument"进行设计的,应该算是 AOP
    的一个支援模块,主要作用是在 JVM 启用时,生成一个代理类,程序员通过代理类在运行时修改类的
    字节,从而改变一个类的功能,实现 AOP 的功能。

6.3、数据访问与集成

由 spring-jdbc、spring-tx、spring-orm、spring-jms 和 spring-oxm 5 个模块组成。

  • spring-jdbc模块是 Spring 提供的 JDBC 抽象框架的主要实现模块,用于简化 Spring JDBC 操作 。
    主要是提供 JDBC 模板方式、关系数据库对象化方式、SimpleJdbc 方式、事务管理来简化 JDBC 编程,
    主要实现类是 JdbcTemplate、SimpleJdbcTemplate 以及 NamedParameterJdbcTemplate。
  • spring-tx 模块是 Spring JDBC 事务控制实现模块。使用 Spring 框架,它对事务做了很好的封装,
    通过它的 AOP 配置,可以灵活的配置在任何一层;但是在很多的需求和应用,直接使用 JDBC 事务控
    制还是有其优势的。其实,事务是以业务逻辑为基础的;一个完整的业务应该对应业务层里的一个方法
    如果业务操作失败,则整个事务回滚;所以,事务控制是绝对应该放在业务层的;但是,持久层的设计
    则应该遵循一个很重要的原则:保证操作的原子性,即持久层里的每个方法都应该是不可以分割的。所
    以,在使用 Spring JDBC 事务控制时,应该注意其特殊性。
  • spring-orm 模块是 ORM 框架支持模块,主要集成 Hibernate, Java Persistence API (JPA) 和
    Java Data Objects (JDO) 用于资源管理、数据访问对象(DAO)的实现和事务策略。
  • spring-oxm 模块主要提供一个抽象层以支撑 OXM(OXM 是 Object-to-XML-Mapping 的缩写,
    它是一个 O/M-mapper,将 java 对象映射成 XML 数据,或者将 XML 数据映射成 java 对象),例如:
    JAXB, Castor, XMLBeans, JiBX 和 XStream 等。
  • spring-jms 模块(Java Messaging Service)能够发送和接收信息,自 Spring Framework 4.1 以
    后,他还提供了对 spring-messaging 模块的支撑

6.4、Web 组件

由 spring-web、spring-webmvc、spring-websocket 和 spring-webflux 4 个模块组成。

  • spring-web 模块为 Spring 提供了最基础 Web 支持,主要建立于核心容器之上,通过 Servlet 或
    者 Listeners 来初始化 IOC 容器,也包含一些与 Web 相关的支持。
  • spring-webmvc 模块众所周知是一个的 Web-Servlet 模块,实现了 Spring MVC
    (model-view-Controller)的 Web 应用。
  • spring-websocket 模块主要是与 Web 前端的全双工通讯的协议。
  • spring-webflux 是一个新的非堵塞函数式 Reactive Web 框架,可以用来建立异步的,非阻塞,
    事件驱动的服务,并且扩展性非常好。

6.5、通信报文

即 spring-messaging 模块,是从 Spring4 开始新加入的一个模块,主要职责是为 Spring 框架集
成一些基础的报文传送应用

6.6、集成测试

即 spring-test 模块,主要为测试提供支持的,毕竟在不需要发布(程序)到你的应用服务器或者连接
到其他企业设施的情况下能够执行一些集成测试或者其他测试对于任何企业都是非常重要的

6.7、集成兼容

即 spring-framework-bom 模块,Bill of Materials.解决 Spring 的不同模块依赖版本不同问题

6.8、各模块之间的依赖关系

在这里插入图片描述
基本学习顺序为:从 spring-core 入手,其次是 spring-beans 和 spring-aop,随后是 spring-context,再其次是 spring-tx 和 spring-orm,最后是 spring-web 和其他部分。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值