在穿越了 Web 容器和Web 應用以後,HTTP 請求將被投送到 Spring 框架,咱們繼續剖析後續流程。Web 應用與 Spring MVC 的銜接是經過配置文件 mvc-servlet.xml 完成的,咱們經過這份配置文件定義構成 Spring MVC 的各類核心組件和初始化配置,其中包括:控制器 Controller、視圖解析器 ViewResolver、視圖 View 等等。不一樣組件分別承擔不一樣的功能,在介紹 Spring 框架處理 HTTP 請求流程以前,咱們照例先了解一下這些核心組件。前端
咱們應用開發者在使用 Spring 時接觸最多的就是各類註解,包括:@Component、@Controller、@Service、@Repository 等,這些都是 Spring 的核心組件。除此以外,咱們還會使用 @RequestMapping、@RequestParam、@PathVariable、@RequestBody 等輔助性註解:數據庫
模型 Model:封裝了業務數據,主要以 POJO 形式存在。後端
控制器 Controller:主要負責具體業務流程的調度控制,以及調用業務邏輯服務 Service。 控制器 Controller 處理完 HTTP 請求以後返回 ModelAndView 對象給前置分發器 DispatcherServlet,ModelAndView 中包含了模型 Model 對象和視圖 View 名稱。前端工程化
視圖 View:負責渲染呈現模型 Model 數據,以及提供表單供用戶錄入各類業務數據。服務器
視圖解析器 ViewResolver:負責根據已知的視圖名稱獲取一個特定視圖 View 子類實例對象。微信
處理器映射 HandlerMapping:主要負責控制器 Controller 的註冊和查找,由應用上下文 ApplicationContext 持有。具體實現上它擁有 HashMap<String, Object> 類型的成員屬性 handlerMap,其中 key 是 HTTP 請求的 URI 信息,value 能夠是一個字符串或者處理請求的 HandlerExecutionChain。若是是 String,則視爲 Spring Bean 名稱。架構
服務 Service:主要負責實現具體的業務邏輯。mvc
數據存儲對象 Repository:也稱爲數據訪問對象 DAO(Data Access Object),無論採用什麼開發框架,大部分應用都須要跟數據庫交互,DAO 就是將訪問數據庫操做作了封裝,隔離了 SQL 相關複雜度。 app
Web 容器監聽主機特定端口,每當有請求抵達時,Web 容器最終將調用 Servlet 的 service 方法處理 HTTP 請求。在 Spring Web 應用中,接收 HTTP 請求的 Servlet 就是前置分發器 DispatcherServlet。框架
在前置分發器 DispatcherServlet 的 service 方法中判斷 HTTP 請求類型,包括:GET、POST、PUT、PATCH 等等,而後再決定調用 doGet()、doPost()、doPut() 等方法。
在 doGet()、doPost()、doPut() 等方法中執行 processRequest() 方法,完成請求上下文的的初始化。
調用 doService() 方法,進一步執行 doDispatch() 方法。
在 doDispatch() 方法中獲取 HTTP 請求的 mappedHandler 和 HandlerAdapter,而後再發起對業務控制器 Controller 的調用以及後續流程,等待處理結果再構建響應數據。
在填充處理器 Handler 入參的過程當中,Spring 還會根據配置作些預處理工做:
Spring Web 應用架構經歷了多個階段的發展,最初主流的前端視圖技術就是 JSP,在此基礎上又演化出了三劍客框架 SSH(Struts\Spring\Hibernate),但這時候先後端其實仍是耦合在一塊兒的,不論是 JSP 仍是 SSH,在前面 Spring 框架處理 HTTP 請求的流程中,必需要依賴視圖解析器 ViewResolver 和視圖 View。
從 Spring 誕生到如今已經15年多了,它關聯的後端技術演化其實沒有前端那麼快,主要緣由就是前端需求愈來愈豐富多樣,前端視圖層的開發工做量和複雜度不斷增長。在這樣的背景之下,愈來愈多的前端工程化解決方案涌現,其中最有成效的就是先後端分離,從 AngularJS\Backbone.js 到如今 React\Vue 等。在這種先後端分離架構下,前端就所有由靜態資源(HTML\Javascript\CSS)等構成,能夠獨立部署在 Web 服務器當中,這樣 Spring 框架就不須要再處理視圖相關的內容,控制器 Controller 再也不返回 ModelAndView,只須要反饋模型數據了。
本系列其餘文章索引以下: