全程手写Spring MVC有多难?一、配置阶段二、初始化阶段三、运行阶段

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

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

image

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

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

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

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

image

一、配置阶段

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

image

然后,创建GPDispatcherServlet类并继承HttpServlet,重写init()、doGet()和doPost()方法。

image

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

image

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

image

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

创建GPController注解:

image

创建GPRequestMapping注解:

image

创建GPService注解:

image

创建GPAutowired注解:

image

创建GPRequestParam注释:

image

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

image

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

二、初始化阶段

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

image

当Servlet容器启动时,会调用GPDispatcherServlet的init()方法,从init方法的参数中,我们可以拿到主配置文件的路径,从能够读取到配置文件中的信息。前面我们已经介绍了Spring的三个阶段,现在来完成初始化阶段的代码。在init()方法中,定义好执行步骤,如下:如果想学习Java工程化、高性能及分布式、深入浅出。微服务、Spring,MyBatis,Netty源码分析的朋友可以加我的Java进阶群:725219329,群里有阿里大牛直播讲解技术,以及Java大型互联网技术的视频免费分享给大家。

image

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

image

doScanner()方法,递归扫描出所有的Class文件

image

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

image

然后,再处理相关的类。

image

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

image

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

image

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

三、运行阶段

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

image

doDispatch()方法是这样写的:

image

到此,我们完成了一个mini版本的Spring,麻雀虽小,五脏俱全。我们把服务发布到web容器中,然后,在浏览器输入:http://localhost:8080/demo/query.json?name=Tom,就会得到下面的结果:如果想学习Java工程化、高性能及分布式、深入浅出。微服务、Spring,MyBatis,Netty源码分析的朋友可以加我的Java进阶群:725219329,群里有阿里大牛直播讲解技术,以及Java大型互联网技术的视频免费分享给大家。

image

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

image

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏一名叫大蕉的程序员

关于Java健壮性的一些思考与实践 No.102

上面两种模式都可以实现标准的 response 的封装,那么具体要封装哪些东西呢?其实最主要的就是统一的 try catch,防止出现任何的 500 错误给到调...

10920
来自专栏数据之美

Nginx log error:client sent invalid userid cookie

基于日志的统计分析按日志来源一般分为后端 cgi、app 日志和前端 js 挂码日志,其中前端 js 挂码由于与具体后端业务逻辑低耦合、异步加载等特性,使得其在...

46990
来自专栏北京马哥教育

干货 | 史上最全的 Python 爬虫工具列表大全

来源:伯乐在线 这个列表包含与网页抓取和数据处理的Python库。 网络 通用 urllib -网络库(stdlib)。 requests -网络库。 gra...

826140
来自专栏Jimoer

JVM学习记录-线程安全与锁优化(二)

高效并发是程序员们写代码时一直所追求的,HotSpot虚拟机开发团队也为此付出了很多努力,为了在线程之间更高效地共享数据,以及解决竞争问题,HotSpot开发团...

10720
来自专栏自由而无用的灵魂的碎碎念

分享:Linux中存储设备的表示

摘要: 硬盘和硬盘分区在Linux都表示为设备,按我们通俗的说法来说,就是怎么来表示或描述硬盘和或硬盘分区,但这种描述应该是科学和具体的;比如IDE硬盘,在Li...

15520
来自专栏玄魂工作室

Hacker基础之工具篇 APT2

所有模块结果都存储在本地主机上,并且是APT2知识库(Knowledge Base)的一部分

12730
来自专栏Netkiller

Plugin Hook 设计与实现

Plugin 跟 Hook 有什么区别,我的理解是编译语言更多使用 Plugin 一词,而动态语言更喜欢使用 Hook 一次。它们的功能都是扩展当前应用软件的功...

37970
来自专栏软件开发

JavaSE学习总结(一)——Java基础

一、Java是什么 Java 是由 Sun Microsystems 在 1995 年首先发布的编程语言和计算平台。Java 是一项用于开发应用程序的技术,可以...

41950
来自专栏非著名程序员

推荐几个非常有用的开发工具之Android Studio插件

推荐几个非常有用的开发工具 非著名程序员 我们都知道Eclipse开发Android将在今年年底google不再继续提供相应的开发支持,转而开始强烈发展Andr...

20380
来自专栏机器学习算法与Python学习

干货 | Python 爬虫的工具列表大全

源 | 伯乐头条 | 小象 这个列表包含与网页抓取和数据处理的Python库。 网络 通用 urllib -网络库(stdlib)。 requests -网络...

35390

扫码关注云+社区

领取腾讯云代金券