人見人愛的Spring已然不單單只是一個框架了。現在,Spring已然成爲了一個生態。但深刻了解Spring的卻寥寥無幾。這裏,我帶你們一塊兒來看看,我是如何手寫Spring的。我將結合對Spring十多年的研究經驗,用不到400行代碼來描述SpringIOC、DI、MVC的精華設計思想,並保證基本功能完整。web
首先,咱們先來介紹一下Spring的三個階段,配置階段、初始化階段和運行階段(如圖):json
配置階段:主要是完成application.xml配置和Annotation配置。api
初始化階段:主要是加載並解析配置信息,而後,初始化IOC容器,完成容器的DI操做,已經完成HandlerMapping的初始化。瀏覽器
運行階段:主要是完成Spring容器啓動之後,完成用戶請求的內部調度,並返回響應結果。app
先來看看咱們的項目結構(以下圖)框架
我採用的是maven管理項目。先來看pom.xml文件中的配置,我只引用了servlet-api的依賴。maven
而後,建立GPDispatcherServlet類並繼承HttpServlet,重寫init()、doGet()和doPost()方法。分佈式
在web.xml文件中配置如下信息:微服務
在<init-param>中,咱們配置了一個初始化加載的Spring主配置文件路徑,在原生框架中,咱們應該配置的是classpath:application.xml。在這裏,咱們爲了簡化操做,用properties文件代替xml文件。如下是properties文件中的內容:工具
接下來,咱們要配置註解。如今,咱們不使用Spring的一針一線,全部註解所有本身手寫。
建立GPController註解:
建立GPRequestMapping註解:
建立GPService註解:
建立GPAutowired註解:
建立GPRequestParam註釋:
使用自定義註解進行配置:
到此,咱們把配置階段的代碼所有手寫完成。
先在GPDispatcherServlet中聲明幾個成員變量:
當Servlet容器啓動時,會調用GPDispatcherServlet的init()方法,從init方法的參數中,咱們能夠拿到主配置文件的路徑,從可以讀取到配置文件中的信息。前面咱們已經介紹了Spring的三個階段,如今來完成初始化階段的代碼。在init()方法中,定義好執行步驟,以下:若是想學習Java工程化、高性能及分佈式、深刻淺出。微服務、Spring,MyBatis,Netty源碼分析的朋友能夠加個人Java進階羣:725219329,羣裏有阿里大牛直播講解技術,以及Java大型互聯網技術的視頻免費分享給你們。
doLoadConfig()方法的實現,將文件讀取到Properties對象中:
doScanner()方法,遞歸掃描出全部的Class文件
doInstance()方法,初始化全部相關的類,並放入到IOC容器之中。IOC容器的key默認是類名首字母小寫,若是是本身設置類名,則優先使用自定義的。所以,要先寫一個針對類名首字母處理的工具方法。
而後,再處理相關的類。
doAutowired()方法,將初始化到IOC容器中的類,須要賦值的字段進行賦值
initHandlerMapping()方法,將GPRequestMapping中配置的信息和Method進行關聯,並保存這些關係。
到此,初始化階段的全部代碼所有寫完。
來到運行階段,當用戶發送請求被Servlet接受時,都會統一調用doPost方法,我先在doPost方法中再調用doDispach()方法,代碼以下:
doDispatch()方法是這樣寫的:
到此,咱們完成了一個mini版本的Spring,麻雀雖小,五臟俱全。咱們把服務發佈到web容器中,而後,在瀏覽器輸入:http://localhost:8080/demo/query.json?name=Tom,就會獲得下面的結果:若是想學習Java工程化、高性能及分佈式、深刻淺出。微服務、Spring,MyBatis,Netty源碼分析的朋友能夠加個人Java進階羣:725219329,羣裏有阿里大牛直播講解技術,以及Java大型互聯網技術的視頻免費分享給你們。
固然,真正的Spring要複雜不少,但核心設計思路基本如此。例如:Spring中真正的HandlerMapping是這樣的: