引言
幾乎每個面試的程序員都會碰到Spring相關(guān)的面試問題,或淺或深。你都是如何回答面試官的問題的?——我不知道,我一般會通過手寫一個Spring來加深自己的印象。
目標(biāo)
人見人愛的Spring已然不僅僅只是一個框架了。如今,Spring已然成為了一個生態(tài)。但深入了解Spring的卻寥寥無幾。這里,我?guī)Т蠹乙黄饋砜纯?,我是如何手寫Spring的。我將結(jié)合對Spring十多年的研究經(jīng)驗,用不到400行代碼來描述SpringIOC、DI、MVC的精華設(shè)計思想,并保證基本功能完整。
開始
首先,我們先來介紹一下Spring的三個階段,配置階段、初始化階段和運行階段(如圖):

配置階段:主要是完成application.xml配置和Annotation配置。
初始化階段:主要是加載并解析配置信息,然后,初始化IOC容器,完成容器的DI操作,已經(jīng)完成HandlerMapping的初始化。
運行階段:主要是完成Spring容器啟動以后,完成用戶請求的內(nèi)部調(diào)度,并返回響應(yīng)結(jié)果。
先來看看我們的項目結(jié)構(gòu)(如下圖)

一、配置階段
我采用的是maven管理項目。先來看pom.xml文件中的配置,我只引用了servlet-api的依賴。

然后,創(chuàng)建GPDispatcherServlet類并繼承HttpServlet,重寫init()、doGet()和doPost()方法。

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

在<init-param>中,我們配置了一個初始化加載的Spring主配置文件路徑,在原生框架中,我們應(yīng)該配置的是classpath:application.xml。在這里,我們?yōu)榱撕喕僮鳎胮roperties文件代替xml文件。以下是properties文件中的內(nèi)容:

接下來,我們要配置注解?,F(xiàn)在,我們不使用Spring的一針一線,所有注解全部自己手寫。
創(chuàng)建GPController注解:

創(chuàng)建GPRequestMapping注解:

創(chuàng)建GPService注解:

創(chuàng)建GPAutowired注解:

創(chuàng)建GPRequestParam注釋:

使用自定義注解進行配置:

到此,我們把配置階段的代碼全部手寫完成。
各位大佬,我有在公開課上有現(xiàn)場直播手寫Spring,歡迎大家關(guān)注??梢约尤海?95244712,了解具體詳情。還有JVM、dubbo、netty等其他知識分享,歡迎加入學(xué)習(xí)。
二、初始化階段
先在GPDispatcherServlet中聲明幾個成員變量:

當(dāng)Servlet容器啟動時,會調(diào)用GPDispatcherServlet的init()方法,從init方法的參數(shù)中,我們可以拿到主配置文件的路徑,從能夠讀取到配置文件中的信息。前面我們已經(jīng)介紹了Spring的三個階段,現(xiàn)在來完成初始化階段的代碼。在init()方法中,定義好執(zhí)行步驟,如下:

doLoadConfig()方法的實現(xiàn),將文件讀取到Properties對象中:

doScanner()方法,遞歸掃描出所有的Class文件

doInstance()方法,初始化所有相關(guān)的類,并放入到IOC容器之中。IOC容器的key默認是類名首字母小寫,如果是自己設(shè)置類名,則優(yōu)先使用自定義的。因此,要先寫一個針對類名首字母處理的工具方法。

然后,再處理相關(guān)的類。

doAutowired()方法,將初始化到IOC容器中的類,需要賦值的字段進行賦值

initHandlerMapping()方法,將GPRequestMapping中配置的信息和Method進行關(guān)聯(lián),并保存這些關(guān)系。

到此,初始化階段的所有代碼全部寫完。
三、運行階段
來到運行階段,當(dāng)用戶發(fā)送請求被Servlet接受時,都會統(tǒng)一調(diào)用doPost方法,我先在doPost方法中再調(diào)用doDispach()方法,代碼如下:

doDispatch()方法是這樣寫的:

各位大佬,我有在公開課上有現(xiàn)場直播手寫Spring,歡迎大家關(guān)注??梢约尤海?95244712,了解具體詳情。還有JVM、dubbo、netty等其他知識分享,歡迎加入學(xué)習(xí)。
收尾
到此,我們完成了一個mini版本的Spring,麻雀雖小,五臟俱全。我們把服務(wù)發(fā)布到web容器中,然后,在瀏覽器輸入:http://localhost:8080/demo/query.json?name=Tom,就會得到下面的結(jié)果:

當(dāng)然,真正的Spring要復(fù)雜很多,但核心設(shè)計思路基本如此。例如:Spring中真正的HandlerMapping是這樣的:

本文轉(zhuǎn)自:http://bbs.gupaoedu.com/forum.php?mod=viewthread&tid=572&tdsourcetag=s_pcqq_aiomsg