人見人愛(ài)的 Spring 已然不僅僅只是一個(gè)框架了。如今,Spring 已然成為了一個(gè)生態(tài)。但深入了解 Spring 的卻寥寥無(wú)幾。這里,我?guī)Т蠹乙黄饋?lái)看看,我是如何手寫 Spring 的。我將結(jié)合對(duì) Spring 十多年的研究經(jīng)驗(yàn),用不到 400 行代碼來(lái)描述 Spring IOC、DI、MVC 的精華設(shè)計(jì)思想,并保證基本功能完整。
首先,我們先來(lái)介紹一下 Spring 的三個(gè)階段,配置階段、初始化階段和運(yùn)行階段(如圖):
配置階段:主要是完成 application.xml 配置和 Annotation 配置。
初始化階段:主要是加載并解析配置信息,然后,初始化 IOC 容器,完成容器的 DI 操作,已經(jīng)完成 HandlerMapping 的初始化。
運(yùn)行階段:主要是完成 Spring 容器啟動(dòng)以后,完成用戶請(qǐng)求的內(nèi)部調(diào)度,并返回響應(yīng)結(jié)果。
先來(lái)看看我們的項(xiàng)目結(jié)構(gòu) (如下圖)
一、配置階段
我采用的是 maven 管理項(xiàng)目。先來(lái)看 pom.xml 文件中的配置,我只引用了 servlet-api 的依賴。
然后,創(chuàng)建 GPDispatcherServlet 類并繼承 HttpServlet,重寫 init()、doGet() 和 doPost() 方法。
在 web.xml 文件中配置以下信息:
在中,我們配置了一個(gè)初始化加載的 Spring 主配置文件路徑,在原生框架中,我們應(yīng)該配置的是 classpath:application.xml。在這里,我們?yōu)榱撕?jiǎn)化操作,用 properties 文件代替 xml 文件。以下是 properties 文件中的內(nèi)容:
接下來(lái),我們要配置注解?,F(xiàn)在,我們不使用 Spring 的一針一線,所有注解全部自己手寫。
創(chuàng)建 GPController 注解:
創(chuàng)建 GPRequestMapping 注解:
創(chuàng)建 GPService 注解:
創(chuàng)建 GPAutowired 注解:
創(chuàng)建 GPRequestParam 注釋:
使用自定義注解進(jìn)行配置:
到此,我們把配置階段的代碼全部手寫完成。
二、初始化階段
先在 GPDispatcherServlet 中聲明幾個(gè)成員變量:
當(dāng) Servlet 容器啟動(dòng)時(shí),會(huì)調(diào)用 GPDispatcherServlet 的 init()方法,從 init 方法的參數(shù)中,我們可以拿到主配置文件的路徑,從能夠讀取到配置文件中的信息。前面我們已經(jīng)介紹了 Spring 的三個(gè)階段,現(xiàn)在來(lái)完成初始化階段的代碼。在 init() 方法中,定義好執(zhí)行步驟,如下:
doLoadConfig() 方法的實(shí)現(xiàn),將文件讀取到 Properties 對(duì)象中:
doScanner() 方法,遞歸掃描出所有的 Class 文件
doInstance() 方法,初始化所有相關(guān)的類,并放入到 IOC 容器之中。IOC 容器的 key 默認(rèn)是類名首字母小寫,如果是自己設(shè)置類名,則優(yōu)先使用自定義的。因此,要先寫一個(gè)針對(duì)類名首字母處理的工具方法。
然后,再處理相關(guān)的類。
doAutowired() 方法,將初始化到 IOC 容器中的類,需要賦值的字段進(jìn)行賦值
initHandlerMapping() 方法,將 GPRequestMapping 中配置的信息和 Method 進(jìn)行關(guān)聯(lián),并保存這些關(guān)系。
到此,初始化階段的所有代碼全部寫完。
三、運(yùn)行階段
來(lái)到運(yùn)行階段,當(dāng)用戶發(fā)送請(qǐng)求被 Servlet 接受時(shí),都會(huì)統(tǒng)一調(diào)用 doPost 方法,我先在 doPost 方法中再調(diào)用 doDispach() 方法,代碼如下:
doDispatch() 方法是這樣寫的:
到此,我們完成了一個(gè) mini 版本的 Spring,麻雀雖小,五臟俱全。我們把服務(wù)發(fā)布到 web 容器中,然后,在瀏覽器輸入:http://localhost:8080/demo/query.json?name=Tom,就會(huì)得到下面的結(jié)果:
當(dāng)然,真正的 Spring 要復(fù)雜很多,但核心設(shè)計(jì)思路基本如此。例如:Spring 中真正的 HandlerMapping 是這樣的:
我在網(wǎng)絡(luò)上也有現(xiàn)場(chǎng)直播手寫 Spring,歡迎大家關(guān)注。如果在練習(xí)過(guò)程中有任何疑問(wèn),可以加我的架構(gòu)群:586446657。
歡迎工作一到五年的 Java 的工程師朋友們加入進(jìn)來(lái),
本群提供免費(fèi)的學(xué)習(xí)指導(dǎo)架構(gòu)資料以及免費(fèi)的解答
不懂得問(wèn)題都可以在本群提出來(lái)之后還會(huì)有職業(yè)生涯規(guī)劃以及面試指導(dǎo)