十年架構師:我是這樣手寫Spring的,用300行代碼體現優雅之道

人見人愛的Spring已然不僅僅只是一個框架了。如今,Spring已然成爲了一個生態。但深入瞭解Spring的卻寥寥無幾。這裏,我帶大家一起來看看,我是如何手寫Spring的。我將結合對Spring十多年的研究經驗,用不到400行代碼來描述SpringIOC、DI、MVC的精華設計思想,並保證基本功能完整。

首先,我們先來介紹一下Spring的三個階段,配置階段、初始化階段和運行階段(如圖):

1240

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

初始化階段:主要是加載並解析配置信息,然後,初始化IOC容器,完成容器的DI操作,已經完成HandlerMapping的初始化。

運行階段:主要是完成Spring容器啓動以後,完成用戶請求的內部調度,並返回響應結果。

先來看看我們的項目結構(如下圖)

1240

一、配置階段

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

1240

然後,創建GPDispatcherServlet類並繼承HttpServlet,重寫init()、doGet()和doPost()方法。

1240

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

1240

在<init-param>中,我們配置了一個初始化加載的Spring主配置文件路徑,在原生框架中,我們應該配置的是classpath:application.xml。在這裏,我們爲了簡化操作,用properties文件代替xml文件。以下是properties文件中的內容:

1240

接下來,我們要配置註解。現在,我們不使用Spring的一針一線,所有註解全部自己手寫。

創建GPController註解:

1240

創建GPRequestMapping註解:

1240

創建GPService註解:

1240

創建GPAutowired註解:

1240

創建GPRequestParam註釋:

1240

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

1240

到此,我們把配置階段的代碼全部手寫完成。

二、初始化階段

先在GPDispatcherServlet中聲明幾個成員變量:

1240

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

1240

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

1240

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

1240

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

1240

然後,再處理相關的類。

1240

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

1240

initHandlerMapping()方法,將GPRequestMapping中配置的信息和Method進行關聯,並保存這些關係。

1240

到此,初始化階段的所有代碼全部寫完。

三、運行階段

來到運行階段,當用戶發送請求被Servlet接受時,都會統一調用doPost方法,我先在doPost方法中再調用doDispach()方法,代碼如下:

1240

doDispatch()方法是這樣寫的:

1240

到此,我們完成了一個mini版本的Spring,麻雀雖小,五臟俱全。我們把服務發佈到web容器中,然後,在瀏覽器輸入:http://localhost:8080/demo/query.json?name=Tom,就會得到下面的結果:

1240

當然,真正的Spring要複雜很多,但核心設計思路基本如此。例如:Spring中真正的HandlerMapping是這樣的:

1240

加架構Java架構圈子:867857579  領取資料,裏面會免費分享一些資深架構師錄製的視頻錄像:有Spring,MyBatis,Netty源碼分析,高併發、高性能、分佈式、微服務架構的原理,JVM性能優化這些成爲架構師必備的資料

1240


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章