Spring 處理過程分析

一、處理過程分析

    1、首先,Tomcat每次啟動時都會加載并解析/WEB-INF/web.xml文件,所以可以先從web.xml找突破口,主要代碼如下:
<servlet ><servlet-name >spring-mvc</servlet-name><!-- servlet類 --><servlet-class >org.springframework.web.servlet.DispatcherServlet</servlet-class><!-- 初始化參數 --><init-param ><param-name >contextConfigLocation</param-name><param-value >classpath:/spring-mvc.xml</param-value></init-param><!-- 啟動時加載 --><load-on-startup >1</load-on-startup></servlet><servlet-mapping ><servlet-name >spring-mvc</servlet-name><url-pattern >/</url-pattern></servlet-mapping>

很幸運,我們可以從web.xml文件獲得三個信息,分別是:servlet類為DispatcherServlet,它在啟動時加載,加載時初始化參數contextConfigLocation?
為classpath下spring-mvc.xml的文件地址,接下來我們將目光移到DispatcherServlet類。

2、打開DispatcherServlet類,我們先將目光聚焦在它的結構體系上,如下圖:?
這里寫圖片描述

很明顯,它是一個Servlet的子類,其實不用說也知道,因為web.xml早已有配置。既然是Servlet,我們就要專注于它的service、doGet、doPost等相關方法,在它的父類FrameServlet,我們找到了service方法。

/*** Override the parent class implementation in order to intercept PATCH* requests.*/@Overrideprotected void service(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {String method = request.getMethod();if (method.equalsIgnoreCase(RequestMethod.PATCH.name())) {processRequest(request, response);}else {super.service(request, response);}}

根據service方法,我們一步步找到一個方法鏈service –> processRequest –> doService –> doDispatch,我們最終將目光定位在doDispatch,因為從它的方法體就可以看出它是整個SpringMVC的核心方法。

protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception {HttpServletRequest processedRequest = request;HandlerExecutionChain mappedHandler = null;boolean multipartRequestParsed = false;WebAsyncManager asyncManager = WebAsyncUtils.getAsyncManager(request);try {ModelAndView mv = null;Exception dispatchException = null;try {//處理文件上傳請求processedRequest = checkMultipart(request);multipartRequestParsed = processedRequest != request;// 解析請求,獲取HandlerExecutionChain對象mappedHandler = getHandler(processedRequest);if (mappedHandler == null || mappedHandler.getHandler() == null) {noHandlerFound(processedRequest, response);return;}// 從HandlerExecutionChain對象獲取HandlerAdapter對象,實際上是從HandlerMapping對象中獲取HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());// Process last-modified header, if supported by the handler.String method = request.getMethod();boolean isGet = "GET".equals(method);if (isGet || "HEAD".equals(method)) {long lastModified = ha.getLastModified(request, mappedHandler.getHandler());if (logger.isDebugEnabled()) {logger.debug("Last-Modified value for [" + getRequestUri(request) + "] is: " + lastModified);}if (new ServletWebRequest(request, response).checkNotModified(lastModified) && isGet) {return;}}//在controller方法執行前,執行攔截器的相關方法(pre)if (!mappedHandler.applyPreHandle(processedRequest, response)) {return;}try {// 執行HandlerAdapter對象的handler方法,返回ModelAndViewmv = ha.handle(processedRequest, response, mappedHandler.getHandler());}finally {if (asyncManager.isConcurrentHandlingStarted()) {return;}}applyDefaultViewName(request, mv);//在controller方法執行后,執行攔截器的相關方法(post)mappedHandler.applyPostHandle(processedRequest, response, mv);}catch (Exception ex) {dispatchException = ex;}//進行視圖解析processDispatchResult(processedRequest, response, mappedHandler, mv, dispatchException);}catch (Exception ex) {triggerAfterCompletion(processedRequest, response, mappedHandler, ex);}catch (Error err) {triggerAfterCompletionWithError(processedRequest, response, mappedHandler, err);}finally {if (asyncManager.isConcurrentHandlingStarted()) {// Instead of postHandle and afterCompletionmappedHandler.applyAfterConcurrentHandlingStarted(processedRequest, response);return;}// Clean up any resources used by a multipart request.if (multipartRequestParsed) {cleanupMultipart(processedRequest);}}}

說它是核心一點也不為過,從上述代碼的中文注釋可以看出,它包含了解析請求,執行相關攔截器,執行handle方法(到這里關于handle方法是什么,我們一臉懵逼。別急,接下來我們會講述,總之它很重要就對了),執行視圖解析方法。?
3、至于HandlerAdapter是干嘛用的?它的handler方法有什么用?我們毫無概念,接下來我們從另一個角度切入(就像兩個人相對而行,一個人筋疲力盡了,唯有靠另一個人努力前行才能相遇),所以我選擇Controller,得先從配置文件入手,因為它采用了spring的IOC。

<bean id="controller" class="com.mvc.controller.MyController"></bean><bean id="interceptor" class="com.mvc.interceptor.MyInterceptor"></bean><bean id="urlMapping" class="org.springframework.web.servlet.handler.SimpleUrlHandlerMapping"><property name="mappings"><props><prop key="controller">controller</prop></props></property><property name="interceptors"><array><ref bean="interceptor"></ref></array></property></bean>

配置文件又給了我們一條重要的信息:controller和攔截器都是作為SimpleUrlHandlerMapping的參數傳進去的,而SimpleUrlHandlerMapping是HandlerMapping的子類。從這里就可以猜測,controller的核心方法要么被HandlerMapping直接調用,要么被HandlerMapping的附屬產品(類)進行調用,接下來我們查看一下controller核心方法的調用情況。?
這里寫圖片描述

很幸運,看到SimpleControllerHandlerAdapter和DispatcherServlet.doDispatch(request, response),我好像發現了新大陸,這不正是我們想要的嗎?HandlerAdapter類和doDispatch(request, response)方法完美地結合在了一起。再看SimpleControllerHandlerAdapter的handler方法:

@Overridepublic ModelAndView handle(HttpServletRequest request, HttpServletResponse response, Object handler)throws Exception {return ((Controller) handler).handleRequest(request, response);}

這里也有一個方法的調用鏈,從上圖就可以看出,handle方法最終是調用handleRequestInternal方法,也就是我們在controller中自定義的方法。總而言之,HandlerAdapter的handler方法是用來調用controller中的handleRequestInternal方法的,而handleRequestInternal的方法體正是我們用戶自定義的業務邏輯。?
好,SpringMVC的主要源碼我們就解析到這里了,接下來我們就SpringMVC的處理過程做一個總結。

二、SpringMVC處理過程總結

    先放一張圖,我們再慢慢解析:

這里寫圖片描述

流程解析:?
1、當request到來時,DispatcherServlet對request進行捕獲,并執行doService方法,繼而執行doDispatch方法。?
2、HandlerMapping解析請求,并且返回HandlerExecutionChain(其中包含controllers和interceptors),然后通過HandlerExecutionChain得到Handler相關類,根據Handler獲取執行它的HandlerAdapter類。?
3、先執行攔截器的pre相關方法,接著執行handler方法,它會調用controller的handleRequestInternal方法(方法體由用戶自定義),最后調用攔截器的post相關方法。?
4、解析handler方法返回的ModelAndView(可以在配置文件中配置ResourceViewResolver,也就是視圖解析器),渲染頁面并response給客戶端。

以上是我對SpringMVC處理流程源碼的分析總結,如有有誤之處,還請各位大神指正。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/387020.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/387020.shtml
英文地址,請注明出處:http://en.pswp.cn/news/387020.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

python全棧開發中級班全程筆記(第二模塊、第四章)(常用模塊導入)

python全棧開發筆記第二模塊 第四章 &#xff1a;常用模塊&#xff08;第二部分&#xff09; 一、os 模塊的 詳解 1、os.getcwd() &#xff1a;得到當前工作目錄&#xff0c;即當前python解釋器所在目錄路徑 import os j os.getcwd() # 返回當前pyt…

基于 Spring Cloud 完整的微服務架構實戰

本項目是一個基于 Spring Boot、Spring Cloud、Spring Oauth2 和 Spring Cloud Netflix 等框架構建的微服務項目。 作者&#xff1a;Sheldon地址&#xff1a;https://github.com/zhangxd1989 技術棧 Spring boot - 微服務的入門級微框架&#xff0c;用來簡化 Spring 應用的初…

mysql Invalid use of group function的解決辦法

錯誤語句&#xff1a;SELECT s.SID, s.Sname, AVG(a.score)FROM student sLEFT JOIN sc a ON s.SID a.SID WHERE AVG(a.score) > 60GROUP BY s.SID正確語句&#xff1a; SELECTs.SID,s.Sname,AVG(a.score)FROMstudent sLEFT JOIN sc a ON s.SID a.SID GROUP BYs.SID HAVIN…

ipython notebook 中 wavefile, display, Audio的使用

基于ipython notebook的 wavefile以及display, Audio的使用首先是使用的庫使用 wavfile 讀取.wav文件使用display,Audio播放聲音最近在做聲音信號處理的時候&#xff0c;使用了ipython notebook。發現相較于matlab&#xff0c;python在有關生成wave文件和播放音頻需要利用到sci…

spring 設計模式

設計模式作為工作學習中的枕邊書&#xff0c;卻時常處于勤說不用的尷尬境地&#xff0c;也不是我們時常忘記&#xff0c;只是一直沒有記憶。 今天&#xff0c;螃蟹在IT學習者網站就設計模式的內在價值做一番探討&#xff0c;并以spring為例進行講解&#xff0c;只有領略了其設計…

Ansible-----循環

with_dict迭代字典項 使用"with_dict"可以迭代字典項。迭代時&#xff0c;使用"item.key"表示字典的key&#xff0c;"item.value"表示字典的值。 ---- hosts: localhosttasks:- debug: msg"{{item.key}} & {{item.value}}"with_di…

ROS(Robot Operating System)筆記 : 1.使用launch file在gazebo中生成urdf機器人

ROS(Robot Operating System) 1.使用launch file在gazebo中生成urdf機器人 最近接觸了ROS(Robot Operating System),發現單單學習官網http://wiki.ros.org/上的教程&#xff0c;在實際操作過程中仍然會遭遇許多困難。這一系列關于ROS的文章記錄了ROS學習過程中可能遇到的問題…

[asp.net] 利用WebClient上傳圖片到遠程服務

一、客戶端 1.頁面 <form id"Form1" method"post" runat"server" enctype"multipart/form-data">     <input id"MyFile" type"file" runat"server" />     <br />     …

ROS project part 1: Ubuntu中安裝opencv包以及相應的依賴

首先在ubuntu linux上安裝opencv $ sudo apt-get install python-opencv使用ipython 驗證 opencv的安裝 $ import cv2 as cv $ print(cv.__version__)查看當前的ubuntu 版本 $ cat /etc/issue查看當前python版本 下列代碼分別用于查看python3 python2的已安裝版本 $ python…

FastReport4.6程序員手冊_翻譯

一、使用TfrxReport 組件工作1、加載并存儲報表默認情況下&#xff0c;報表窗體同項目窗體構存儲在同一個DFM文件中。多數情況下&#xff0c;無須再操作&#xff0c;因而你就不必采用特殊方法加載報表。如果你決定在文件中存儲報表窗體或者是數據庫的Blob字段&#xff08;他提供…

Python音頻信號處理 1.短時傅里葉變換及其逆變換

短時傅里葉變換及其逆變換 本篇文章主要記錄了使用python進行短時傅里葉變換&#xff0c;分析頻譜&#xff0c;以及通過頻譜實現在頻域內降低底噪的代碼及分析&#xff0c;希望可以給同樣在學習信號處理的大家一點幫助&#xff0c;也希望大家對我的文章多提意見建議。 一. 短…

Java多線程同步機制

一段synchronized的代碼被一個線程執行之前&#xff0c;他要先拿到執行這段代碼的權限&#xff0c;在 java里邊就是拿到某個同步對象的鎖&#xff08;一個對象只有一把鎖&#xff09;&#xff1b; 如果這個時候同步對象的鎖被其他線程拿走了&#xff0c;他&#xff08;這個線程…

SpringBoot與數據訪問

pom依賴&#xff1a; <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-jdbc</artifactId></dependency><dependency><groupId>mysql</groupId><artifactId>mysql-conne…

Python音頻信號處理 2.使用譜減法去除音頻底噪

使用譜減法去除音頻底噪 上一篇文章我主要分享了短時傅立葉變換及其逆變換在python中的實現&#xff0c;有興趣的可以閱讀一下該篇文章&#xff0c;地址如下&#xff1a; Python音頻信號處理 1.短時傅里葉變換及其逆變換 那么在本篇文章中&#xff0c;我們將利用短時傅立葉變…

線程池的優點

線程池的優點 1、線程是稀缺資源&#xff0c;使用線程池可以減少創建和銷毀線程的次數&#xff0c;每個工作線程都可以重復使用。 2、可以根據系統的承受能力&#xff0c;調整線程池中工作線程的數量&#xff0c;防止因為消耗過多內存導致服務器崩潰。 線程池的創建 public…

ROS(Robot Operating System)筆記 : 2.創建并配置package

ROS(Robot Operating System)筆記 : 2.創建一個ROS包并設置其依賴 1.首先來到ros的工作目錄下&#xff0c;接著使用 catkin_make [包名稱] [依賴1] [依賴2] … 創建一個包名為 challenge_project 的 ros包。 $ catkin_create_pkg challenge_project rospy std_msgs cv_bri…

Java線程相關的熱門面試題

1) 什么是線程&#xff1f; 線程是操作系統能夠進行運算調度的最小單位&#xff0c;它被包含在進程之中&#xff0c;是進程中的實際運作單位。程序員可以通過它進行多處理器編程&#xff0c;你可以使用多線程對運算密集型任務提速。比如&#xff0c;如果一個線程完成一個任務要…

linux運維、架構之路-jumpserver

linux運維、架構之路-jumpserver 一、jumpserver介紹 是一款由python編寫開源的跳板機(堡壘機)系統&#xff0c;實現了跳板機應有的功能。基于ssh協議來管理&#xff0c;客戶端無需安裝agent。 特點&#xff1a; 完全開源&#xff0c;GPL授權 Python編寫&#xff0c;容易再次開…

C++ STL學習筆記 : 1. template 模板函數

本篇文章是學習C STL庫的第一篇筆記&#xff0c;主要記錄了使用template關鍵字創建模板函數的方法。 下面用一個非常簡單的例子解釋模板函數的用法 : #include <iostream> using namespace std;template <class T> void myswap(T& a, T& b) {T temp a;a…

C++ STL學習筆記 : 2. unordered map 容器

本文中&#xff0c;簡單總結一下使用unordered map 的心得。unordered_map容器屬于STL中關聯表的一種&#xff0c;常用的map容器與unordered_map容器在使用中有著很大程度的相同點&#xff0c;在之后的文章中我可能會針對二者的相同點與不同點進行細致的分析&#xff0c;這里就…