阿里顶级Java架构师,教你这样手写Spring!

人见人爱的Spring已然不单单只是一个框架了。现在,Spring已然成为了一个生态。但深刻了解Spring的却寥寥无几。这里,我带你们一块儿来看看,我是如何手写Spring的。我将结合对Spring十多年的研究经验,用不到400行代码来描述SpringIOC、DI、MVC的精华设计思想,并保证基本功能完整。web

首先,咱们先来介绍一下Spring的三个阶段,配置阶段、初始化阶段和运行阶段(如图):json

配置阶段:主要是完成application.xml配置和Annotation配置。api

初始化阶段:主要是加载并解析配置信息,而后,初始化IOC容器,完成容器的DI操做,已经完成HandlerMapping的初始化。浏览器

运行阶段:主要是完成Spring容器启动之后,完成用户请求的内部调度,并返回响应结果。架构

先来看看咱们的项目结构(以下图)app

1、配置阶段框架

我采用的是maven管理项目。先来看pom.xml文件中的配置,我只引用了servlet-api的依赖。maven

而后,建立GPDispatcherServlet类并继承HttpServlet,重写init()、doGet()和doPost()方法。分布式

在web.xml文件中配置如下信息:工具

在<init-param>中,咱们配置了一个初始化加载的Spring主配置文件路径,在原生框架中,咱们应该配置的是classpath:application.xml。在这里,咱们为了简化操做,用properties文件代替xml文件。如下是properties文件中的内容:

接下来,咱们要配置注解。如今,咱们不使用Spring的一针一线,全部注解所有本身手写。

建立GPController注解:

建立GPRequestMapping注解:

建立GPService注解:

建立GPAutowired注解:

建立GPRequestParam注释:

使用自定义注解进行配置:

到此,咱们把配置阶段的代码所有手写完成。

2、初始化阶段

先在GPDispatcherServlet中声明几个成员变量:

当Servlet容器启动时,会调用GPDispatcherServlet的init()方法,从init方法的参数中,咱们能够拿到主配置文件的路径,从可以读取到配置文件中的信息。前面咱们已经介绍了Spring的三个阶段,如今来完成初始化阶段的代码。在init()方法中,定义好执行步骤,以下:

doLoadConfig()方法的实现,将文件读取到Properties对象中:

doScanner()方法,递归扫描出全部的Class文件

doInstance()方法,初始化全部相关的类,并放入到IOC容器之中。IOC容器的key默认是类名首字母小写,若是是本身设置类名,则优先使用自定义的。所以,要先写一个针对类名首字母处理的工具方法。

而后,再处理相关的类。

doAutowired()方法,将初始化到IOC容器中的类,须要赋值的字段进行赋值

initHandlerMapping()方法,将GPRequestMapping中配置的信息和Method进行关联,并保存这些关系。

到此,初始化阶段的全部代码所有写完。

3、运行阶段

来到运行阶段,当用户发送请求被Servlet接受时,都会统一调用doPost方法,我先在doPost方法中再调用doDispach()方法,代码以下:

doDispatch()方法是这样写的:

到此,咱们完成了一个mini版本的Spring,麻雀虽小,五脏俱全。咱们把服务发布到web容器中,而后,在浏览器输入:http://localhost:8080/demo/query.json?name=Tom,就会获得下面的结果:

固然,真正的Spring要复杂不少,但核心设计思路基本如此。例如:Spring中真正的HandlerMapping是这样的:

我在公开课上有现场直播手写Spring,欢迎你们关注。针对上面的技术我特地整理了一下,有不少技术不是靠几句话能讲清楚,因此干脆找朋友录制了一些视频,不少问题其实答案很简单,可是背后的思考和逻辑不简单,要作到知其然还要知其因此然。若是想学习Java工程化、高性能及分布式、深刻浅出。性能调优、Spring,MyBatis,Netty源码分析的朋友能够来个人Java 架构开发圈子群:697--579-751