欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

我是这样手写Spring的,麻雀虽小五脏俱全

程序员文章站 2022-06-03 17:59:34
人见人爱的Spring已然不仅仅只是一个框架了。如今,Spring已然成为了一个生态。但深入了解Spring的却寥寥无几。这里,我带大家一起来看看,我是如何手写Spring的。我将结合对Spring十多年的研究经验,用不到400行代码来描述SpringIOC、DI、MVC的精华设计思想,并保证基本功 ......

人见人爱的spring已然不仅仅只是一个框架了。如今,spring已然成为了一个生态。但深入了解spring的却寥寥无几。这里,我带大家一起来看看,我是如何手写spring的。我将结合对spring十多年的研究经验,用不到400行代码来描述springioc、di、mvc的精华设计思想,并保证基本功能完整。

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

我是这样手写Spring的,麻雀虽小五脏俱全

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

初始化阶段:主要是加载并解析配置信息,然后,初始化ioc容器,完成容器的di操作,已经完成handlermapping的初始化。

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

先来看看我们的项目结构(如下图)

我是这样手写Spring的,麻雀虽小五脏俱全

一、配置阶段

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

我是这样手写Spring的,麻雀虽小五脏俱全

然后,创建gpdispatcherservlet类并继承httpservlet,重写init()、doget()和dopost()方法。

我是这样手写Spring的,麻雀虽小五脏俱全

在web.xml文件中配置以下信息:

我是这样手写Spring的,麻雀虽小五脏俱全

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

我是这样手写Spring的,麻雀虽小五脏俱全

接下来,我们要配置注解。现在,我们不使用spring的一针一线,所有注解全部自己手写。

创建gpcontroller注解:

我是这样手写Spring的,麻雀虽小五脏俱全

创建gprequestmapping注解:

我是这样手写Spring的,麻雀虽小五脏俱全

创建gpservice注解:

我是这样手写Spring的,麻雀虽小五脏俱全

创建gpautowired注解:

我是这样手写Spring的,麻雀虽小五脏俱全

创建gprequestparam注释:

我是这样手写Spring的,麻雀虽小五脏俱全

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

我是这样手写Spring的,麻雀虽小五脏俱全

到此,我们把配置阶段的代码全部手写完成。

二、初始化阶段

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

我是这样手写Spring的,麻雀虽小五脏俱全

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

我是这样手写Spring的,麻雀虽小五脏俱全

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

我是这样手写Spring的,麻雀虽小五脏俱全

doscanner()方法,递归扫描出所有的class文件

我是这样手写Spring的,麻雀虽小五脏俱全

doinstance()方法,初始化所有相关的类,并放入到ioc容器之中。ioc容器的key默认是类名首字母小写,如果是自己设置类名,则优先使用自定义的。因此,要先写一个针对类名首字母处理的工具方法。

我是这样手写Spring的,麻雀虽小五脏俱全

然后,再处理相关的类。

我是这样手写Spring的,麻雀虽小五脏俱全

doautowired()方法,将初始化到ioc容器中的类,需要赋值的字段进行赋值

我是这样手写Spring的,麻雀虽小五脏俱全

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

我是这样手写Spring的,麻雀虽小五脏俱全

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

三、运行阶段

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

我是这样手写Spring的,麻雀虽小五脏俱全

dodispatch()方法是这样写的:

我是这样手写Spring的,麻雀虽小五脏俱全

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

我是这样手写Spring的,麻雀虽小五脏俱全

当然,真正的spring要复杂很多,但核心设计思路基本如此。例如:spring中真正的handlermapping是这样的:

我是这样手写Spring的,麻雀虽小五脏俱全