Servlet原理 三:Servlet體系結(jié)構(gòu)以及工作原理

注:在網(wǎng)上看到一篇文章--《Servlet工作原理》,整理并做了一些筆記。

1. Servlet 體系結(jié)構(gòu)

Servlet體系結(jié)構(gòu).png

我們知道 Java Web 應(yīng)用是基于 Servlet 規(guī)范運(yùn)轉(zhuǎn)的编检,那么 Servlet 本身又是如何運(yùn)轉(zhuǎn)的呢乏悄?為何要設(shè)計(jì)這樣的體系結(jié)構(gòu)劝篷。


Servlet頂層類.jpg

從上圖可以看出 Servlet 規(guī)范就是基于這幾個(gè)類運(yùn)轉(zhuǎn)的,與 Servlet 主動(dòng)關(guān)聯(lián)的是三個(gè)類,分別是 ServletConfig、ServletRequest 和 ServletResponse瞬痘。這三個(gè)類都是通過容器傳遞給 Servlet 的故慈,其中 ServletConfig 是在 Servlet 初始化時(shí)就傳給 Servlet 了,而后兩個(gè)是在請(qǐng)求達(dá)到時(shí)調(diào)用 Servlet 時(shí)傳遞過來的框全。

我們應(yīng)該很清楚 ServletRequest 和 ServletResponse 在 Servlet 運(yùn)行的意義察绷,但是 ServletConfig 和 ServletContext 對(duì) Servlet 有何價(jià)值?

仔細(xì)查看 ServletConfig 接口中聲明的方法就會(huì)發(fā)現(xiàn)津辩,這些方法都是為了獲取這個(gè) Servlet 的一些配置屬性拆撼,而這些配置屬性可能會(huì)在 Servlet 運(yùn)行時(shí)被用到。

那么 ServletContext 又是干什么的呢喘沿? Servlet 的運(yùn)行模式是一個(gè)典型的“握手型的交互式”運(yùn)行模式闸度。所謂“握手型的交互式”就是兩個(gè)模塊為了交換數(shù)據(jù)通常都會(huì)準(zhǔn)備一個(gè)交易場(chǎng)景,這個(gè)場(chǎng)景一直跟隨個(gè)這個(gè)交易過程直到這個(gè)交易完成為止蚜印。這個(gè)交易場(chǎng)景的初始化是根據(jù)這次交易對(duì)象指定的參數(shù)來定制的莺禁,這些指定參數(shù)通常就會(huì)是一個(gè)配置類。所以對(duì)號(hào)入座窄赋,交易場(chǎng)景就由 ServletContext 來描述睁宰,而定制的參數(shù)集合就由 ServletConfig 來描述。而 ServletRequest 和 ServletResponse 就是要交互的具體對(duì)象了寝凌,它們通常都是作為運(yùn)輸工具來傳遞交互結(jié)果。

ServletConfig 是在 Servlet init 時(shí)由容器傳過來的孝赫,那么 ServletConfig 到底是個(gè)什么對(duì)象呢较木?


ServletConfig.jpg

上圖可以看出 StandardWrapper 和 StandardWrapperFacade 都實(shí)現(xiàn)了 ServletConfig 接口,而 StandardWrapperFacade 是 StandardWrapper 門面類青柄,所以傳給 Servlet 的是 StandardWrapperFacade 對(duì)象伐债,這個(gè)類能夠保證從 StandardWrapper 中拿到 ServletConfig 所規(guī)定的數(shù)據(jù),而又不把 ServletConfig 不關(guān)心的數(shù)據(jù)暴露給 Servlet致开。

同樣 ServletContext 也與 ServletConfig 有類似的結(jié)構(gòu)峰锁,Servlet 中能拿到的 ServletContext 的實(shí)際對(duì)象也是 ApplicationContextFacade 對(duì)象。ApplicationContextFacade 同樣保證 ServletContex 只能從容器中拿到它該拿的數(shù)據(jù)双戳,它們都起到對(duì)數(shù)據(jù)的封裝作用虹蒋,它們使用的都是門面設(shè)計(jì)模式。

通過 ServletContext 可以拿到 Context 容器中一些必要信息飒货,比如應(yīng)用的工作路徑魄衅,容器支持的 Servlet 最小版本等。

Servlet 中定義的兩個(gè) ServletRequest 和 ServletResponse 它們實(shí)際的對(duì)象又是什么呢塘辅?晃虫,我們?cè)趧?chuàng)建自己的 Servlet 類時(shí)通常使用的都是 HttpServletRequest 和 HttpServletResponse,它們繼承了 ServletRequest 和 ServletResponse扣墩。為何 Context 容器傳過來的 ServletRequest哲银、ServletResponse 可以被轉(zhuǎn)化為 HttpServletRequest 和 HttpServletResponse 呢扛吞?


Request.jpg

上圖是 Tomcat 創(chuàng)建的 Request 和 Response 的類結(jié)構(gòu)圖。Tomcat 一接受到請(qǐng)求首先將會(huì)創(chuàng)建 org.apache.coyote.Request 和 org.apache.coyote.Response荆责,這兩個(gè)類是 Tomcat 內(nèi)部使用的描述一次請(qǐng)求和相應(yīng)的信息類它們是一個(gè)輕量級(jí)的類滥比,它們作用就是在服務(wù)器接收到請(qǐng)求后,經(jīng)過簡(jiǎn)單解析將這個(gè)請(qǐng)求快速的分配給后續(xù)線程去處理草巡,所以它們的對(duì)象很小守呜,很容易被 JVM 回收。接下去當(dāng)交給一個(gè)用戶線程去處理這個(gè)請(qǐng)求時(shí)又創(chuàng)建 org.apache.catalina.connector. Request 和 org.apache.catalina.connector. Response 對(duì)象山憨。這兩個(gè)對(duì)象一直穿越整個(gè) Servlet 容器直到要傳給 Servlet查乒,傳給 Servlet 的是 Request 和 Response 的門面類 RequestFacade 和 RequestFacade,這里使用門面模式與前面一樣都是基于同樣的目的——封裝容器中的數(shù)據(jù)郁竟。一次請(qǐng)求對(duì)應(yīng)的 Request 和 Response 的類轉(zhuǎn)化如下圖所示:


Request和Response的轉(zhuǎn)變.jpg

2. Servlet 如何工作

Servlet如何讓工作.png

我們已經(jīng)清楚了 Servlet 是如何被加載的玛迄、Servlet 是如何被初始化的,以及 Servlet 的體系結(jié)構(gòu)棚亩,現(xiàn)在的問題就是它是如何被調(diào)用的蓖议。

當(dāng)用戶從瀏覽器向服務(wù)器發(fā)起一個(gè)請(qǐng)求,通常會(huì)包含如下信息:http://hostname: port /contextpath/servletpath讥蟆,hostname 和 port 是用來與服務(wù)器建立 TCP 連接勒虾,而后面的 URL 才是用來選擇服務(wù)器中那個(gè)子容器服務(wù)用戶的請(qǐng)求。那服務(wù)器是如何根據(jù)這個(gè) URL 來達(dá)到正確的 Servlet 容器中的呢瘸彤?

Tomcat7.0 中這件事很容易解決修然,因?yàn)檫@種映射工作有專門一個(gè)類來完成的,這個(gè)就是 org.apache.tomcat.util.http.mapper质况,這個(gè)類保存了 Tomcat 的 Container 容器中的所有子容器的信息愕宋,當(dāng) org.apache.catalina.connector. Request 類在進(jìn)入 Container 容器之前,mapper 將會(huì)根據(jù)這次請(qǐng)求的 hostnane 和 contextpath 將 host 和 context 容器設(shè)置到 Request 的 mappingData 屬性中结榄。所以當(dāng) Request 進(jìn)入 Container 容器之前中贝,它要訪問那個(gè)子容器這時(shí)就已經(jīng)確定了。


Request的Mapper.jpg

可能你有疑問臼朗,mapper 中怎么會(huì)有容器的完整關(guān)系邻寿,這要回到圖 2 中 19 步 MapperListener 類的初始化過程,下面是 MapperListener 的 init 方法代碼 :

public void init() { 
        findDefaultHost(); 
        Engine engine = (Engine) connector.getService().getContainer(); 
        engine.addContainerListener(this); 
        Container[] conHosts = engine.findChildren(); 
        for (Container conHost : conHosts) { 
            Host host = (Host) conHost; 
            if (!LifecycleState.NEW.equals(host.getState())) { 
                host.addLifecycleListener(this); 
                registerHost(host); 
            } 
        } 
 }

這段代碼的作用就是將 MapperListener 類作為一個(gè)監(jiān)聽者加到整個(gè) Container 容器中的每個(gè)子容器中依溯,這樣只要任何一個(gè)容器發(fā)生變化老厌,MapperListener 都將會(huì)被通知,相應(yīng)的保存容器關(guān)系的 MapperListener 的 mapper 屬性也會(huì)修改黎炉。for 循環(huán)中就是將 host 及下面的子容器注冊(cè)到 mapper 中枝秤。


Request在容器中得路由圖.jpg

上圖描述了一次 Request 請(qǐng)求是如何達(dá)到最終的 Wrapper 容器的,我們現(xiàn)正知道了請(qǐng)求是如何達(dá)到正確的 Wrapper 容器慷嗜,但是請(qǐng)求到達(dá)最終的 Servlet 還要完成一些步驟淀弹,必須要執(zhí)行 Filter 鏈丹壕,以及要通知你在 web.xml 中定義的 listener。

接下去就要執(zhí)行 Servlet 的 service 方法了薇溃,通常情況下菌赖,我們自己定義的 servlet 并不是直接去實(shí)現(xiàn) javax.servlet.servlet 接口,而是去繼承更簡(jiǎn)單的 HttpServlet 類或者 GenericServlet 類沐序,我們可以有選擇的覆蓋相應(yīng)方法去實(shí)現(xiàn)我們要完成的工作琉用。

Servlet 的確已經(jīng)能夠幫我們完成所有的工作了,但是現(xiàn)在的 web 應(yīng)用很少有直接將交互全部頁面都用 servlet 來實(shí)現(xiàn)策幼,而是采用更加高效的 MVC 框架來實(shí)現(xiàn)邑时。這些 MVC 框架基本的原理都是將所有的請(qǐng)求都映射到一個(gè) Servlet,然后去實(shí)現(xiàn) service 方法特姐,這個(gè)方法也就是 MVC 框架的入口晶丘。

當(dāng) Servlet 從 Servlet 容器中移除時(shí),也就表明該 Servlet 的生命周期結(jié)束了唐含,這時(shí) Servlet 的 destroy 方法將被調(diào)用浅浮,做一些掃尾工作。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末捷枯,一起剝皮案震驚了整個(gè)濱河市滚秩,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌淮捆,老刑警劉巖叔遂,帶你破解...
    沈念sama閱讀 206,968評(píng)論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異争剿,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)痊末,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,601評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門蚕苇,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人凿叠,你說我怎么就攤上這事涩笤。” “怎么了盒件?”我有些...
    開封第一講書人閱讀 153,220評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵蹬碧,是天一觀的道長。 經(jīng)常有香客問我炒刁,道長恩沽,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,416評(píng)論 1 279
  • 正文 為了忘掉前任翔始,我火速辦了婚禮罗心,結(jié)果婚禮上里伯,老公的妹妹穿的比我還像新娘。我一直安慰自己渤闷,他們只是感情好疾瓮,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,425評(píng)論 5 374
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著飒箭,像睡著了一般狼电。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上弦蹂,一...
    開封第一講書人閱讀 49,144評(píng)論 1 285
  • 那天肩碟,我揣著相機(jī)與錄音,去河邊找鬼盈匾。 笑死腾务,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的削饵。 我是一名探鬼主播岩瘦,決...
    沈念sama閱讀 38,432評(píng)論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼窿撬!你這毒婦竟也來了启昧?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,088評(píng)論 0 261
  • 序言:老撾萬榮一對(duì)情侶失蹤劈伴,失蹤者是張志新(化名)和其女友劉穎密末,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體跛璧,經(jīng)...
    沈念sama閱讀 43,586評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡严里,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,028評(píng)論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了追城。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片刹碾。...
    茶點(diǎn)故事閱讀 38,137評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖座柱,靈堂內(nèi)的尸體忽然破棺而出迷帜,到底是詐尸還是另有隱情,我是刑警寧澤色洞,帶...
    沈念sama閱讀 33,783評(píng)論 4 324
  • 正文 年R本政府宣布戏锹,位于F島的核電站,受9級(jí)特大地震影響火诸,放射性物質(zhì)發(fā)生泄漏锦针。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,343評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望伞插。 院中可真熱鬧割粮,春花似錦、人聲如沸媚污。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,333評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽耗美。三九已至京髓,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間商架,已是汗流浹背堰怨。 一陣腳步聲響...
    開封第一講書人閱讀 31,559評(píng)論 1 262
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留蛇摸,地道東北人备图。 一個(gè)月前我還...
    沈念sama閱讀 45,595評(píng)論 2 355
  • 正文 我出身青樓,卻偏偏與公主長得像赶袄,于是被迫代替她去往敵國和親揽涮。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,901評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容