Servlet生命周期與工作原理詳解
本文為大家分享了Servlet生命周期與工作原理,供大家參考,具體內(nèi)容如下
Servlet生命周期分為三個(gè)階段:
1、初始化階段 調(diào)用init()方法
2、響應(yīng)客戶請(qǐng)求階段 調(diào)用service()方法
3、終止階段 調(diào)用destroy()方法
Servlet初始化階段:
在下列時(shí)刻Servlet容器裝載Servlet:
1.Servlet容器啟動(dòng)時(shí)自動(dòng)裝載某些Servlet,實(shí)現(xiàn)它只需要在web.XML文件中的<Servlet></Servlet>之間添加如下代碼:
<loadon-startup>1</loadon-startup>
2.在Servlet容器啟動(dòng)后,客戶首次向Servlet發(fā)送請(qǐng)求
3.Servlet類文件被更新后,重新裝載Servlet
Servlet被裝載后,Servlet容器創(chuàng)建一個(gè)Servlet實(shí)例并且調(diào)用Servlet的init()方法進(jìn)行初始化。在Servlet的整個(gè)生命周期內(nèi),init()方法只被調(diào)用一次。
Servlet工作原理:
首先簡(jiǎn)單解釋一下Servlet接收和響應(yīng)客戶請(qǐng)求的過(guò)程,首先客戶發(fā)送一個(gè)請(qǐng)求,Servlet是調(diào)用service()方法對(duì)請(qǐng)求進(jìn)行響應(yīng)的,通過(guò)源代碼可見(jiàn),service()方法中對(duì)請(qǐng)求的方式進(jìn)行了匹配,選擇調(diào)用doGet,doPost等這些方法,然后再進(jìn)入對(duì)應(yīng)的方法中調(diào)用邏輯層的方法,實(shí)現(xiàn)對(duì)客戶的響應(yīng)。在Servlet接口和GenericServlet中是沒(méi)有doGet,doPost等等這些方法的,HttpServlet中定義了這些方法,但是都是返回error信息,所以,我們每次定義一個(gè)Servlet的時(shí)候,都必須實(shí)現(xiàn)doGet或doPost等這些方法。
每一個(gè)自定義的Servlet都必須實(shí)現(xiàn)Servlet的接口,Servlet接口中定義了五個(gè)方法,其中比較重要的三個(gè)方法涉及到Servlet的生命周期,分別是上文提到的init(),service(),destroy()方法。GenericServlet是一個(gè)通用的,不特定于任何協(xié)議的Servlet,它實(shí)現(xiàn)了Servlet接口。而HttpServlet繼承于GenericServlet,因此HttpServlet也實(shí)現(xiàn)了Servlet接口。所以我們定義Servlet的時(shí)候只需要繼承HttpServlet即可。
Servlet接口和GenericServlet是不特定于任何協(xié)議的,而HttpServlet是特定于HTTP協(xié)議的類,所以HttpServlet中實(shí)現(xiàn)了service()方法,并將請(qǐng)求ServletRequest,ServletResponse強(qiáng)轉(zhuǎn)為HttpRequest和HttpResponse。
public void service(ServletRequest req,ServletResponse res) throws ServletException,IOException { HttpRequest request; HttpResponse response; try { req = (HttpRequest)request; res = (HttpResponse)response; }catch(ClassCastException e) { throw new ServletException("non-HTTP request response"); } service(request,response); }
代碼的最后調(diào)用了HTTPServlet自己的service(request,response)方法,然后根據(jù)請(qǐng)求去調(diào)用對(duì)應(yīng)的doXXX方法,因?yàn)镠ttpServlet中的doXXX方法都是返回錯(cuò)誤信息
protected void doGet(HttpServletRequest res,HttpServletResponse resp) throws ServletException,IOException { String protocol = req.getProtocol(); String msg = IStrings.getString("http.method_get_not_supported"); if(protocol.equals("1.1")) { resp.sendError(HttpServletResponse.SC.METHOD.NOT.ALLOWED,msg); } esle { resp.sendError(HttpServletResponse.SC_BAD_REQUEST,msg); } }
所以需要我們?cè)谧远x的Servlet中override這些方法!
源碼面前,了無(wú)秘密!
Servlet響應(yīng)請(qǐng)求階段:
對(duì)于用戶到達(dá)Servlet的請(qǐng)求,Servlet容器會(huì)創(chuàng)建特定于這個(gè)請(qǐng)求的ServletRequest對(duì)象和ServletResponse對(duì)象,然后調(diào)用Servlet的service方法。service方法從ServletRequest對(duì)象獲得客戶請(qǐng)求信息,處理該請(qǐng)求,并通過(guò)ServletResponse對(duì)象向客戶返回響應(yīng)信息。
對(duì)于Tomcat來(lái)說(shuō),它會(huì)將傳遞過(guò)來(lái)的參數(shù)放在一個(gè)Hashtable中,該Hashtable的定義是:
這是一個(gè)String-->String[]的鍵值映射。
HashMap線程不安全的,Hashtable線程安全。
Servlet終止階段:
當(dāng)WEB應(yīng)用被終止,或Servlet容器終止運(yùn)行,或Servlet容器重新裝載Servlet新實(shí)例時(shí),Servlet容器會(huì)先調(diào)用Servlet的destroy()方法,在destroy()方法中可以釋放掉Servlet所占用的資源。
Servlet何時(shí)被創(chuàng)建:
1.默認(rèn)情況下,當(dāng)WEB客戶第一次請(qǐng)求訪問(wèn)某個(gè)Servlet的時(shí)候,WEB容器將創(chuàng)建這個(gè)Servlet的實(shí)例。
2.當(dāng)web.xml文件中如果<servlet>元素中指定了<load-on-startup>子元素時(shí),Servlet容器在啟動(dòng)web服務(wù)器時(shí),將按照順序創(chuàng)建并初始化Servlet對(duì)象。
注意:在web.xml文件中,某些Servlet只有<serlvet>元素,沒(méi)有<servlet-mapping>元素,這樣我們無(wú)法通過(guò)url的方式訪問(wèn)這些Servlet,這種Servlet通常會(huì)在<servlet>元素中配置一個(gè)<load-on-startup>子元素,讓容器在啟動(dòng)的時(shí)候自動(dòng)加載這些Servlet并調(diào)用init()方法,完成一些全局性的初始化工作。
Web應(yīng)用何時(shí)被啟動(dòng):
1.當(dāng)Servlet容器啟動(dòng)的時(shí)候,所有的Web應(yīng)用都會(huì)被啟動(dòng)
2.控制器啟動(dòng)web應(yīng)用
Servlet與JSP的比較:
有許多相似之處,都可以生成動(dòng)態(tài)網(wǎng)頁(yè)。
JSP的優(yōu)點(diǎn)是擅長(zhǎng)于網(wǎng)頁(yè)制作,生成動(dòng)態(tài)頁(yè)面比較直觀,缺點(diǎn)是不容易跟蹤與排錯(cuò)。
Servlet是純Java語(yǔ)言,擅長(zhǎng)于處理流程和業(yè)務(wù)邏輯,缺點(diǎn)是生成動(dòng)態(tài)網(wǎng)頁(yè)不直觀。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Mybatis實(shí)現(xiàn)分頁(yè)的注意點(diǎn)
Mybatis提供了強(qiáng)大的分頁(yè)攔截實(shí)現(xiàn),可以完美的實(shí)現(xiàn)分功能。下面小編給大家分享小編在使用攔截器給mybatis進(jìn)行分頁(yè)所遇到的問(wèn)題及注意點(diǎn),需要的朋友一起看看吧2017-07-07解析MyBatis源碼實(shí)現(xiàn)自定義持久層框架
這篇文章主要介紹了手撕MyBatis源碼實(shí)現(xiàn)自定義持久層框架,涉及到的設(shè)計(jì)模式有Builder構(gòu)建者模式、??模式、代理模式,本文通過(guò)示例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下2022-05-05Spring Boot 訪問(wèn)安全之認(rèn)證和鑒權(quán)詳解
這篇文章主要介紹了Spring Boot 訪問(wèn)安全之認(rèn)證和鑒權(quán),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-11-11SpringBoot?@DS注解實(shí)現(xiàn)多數(shù)據(jù)源配置以及問(wèn)題解決辦法
這篇文章主要給大家介紹了關(guān)于SpringBoot?@DS注解實(shí)現(xiàn)多數(shù)據(jù)源配置以及問(wèn)題解決辦法,所謂多數(shù)據(jù)源就是一個(gè)Java EE項(xiàng)目中采用了不同數(shù)據(jù)庫(kù)實(shí)例中的多個(gè)庫(kù),或者是同一個(gè)數(shù)據(jù)庫(kù)實(shí)例中的多個(gè)不同庫(kù),需要的朋友可以參考下2023-11-11Idea中maven無(wú)法下載依賴包問(wèn)題解決
用過(guò)idea開(kāi)發(fā)過(guò)項(xiàng)目的同學(xué),偶爾會(huì)遇到項(xiàng)目中有一些依賴沒(méi)法下載,或者依賴包已經(jīng)有項(xiàng)目卻無(wú)法掃到的問(wèn)題,本文就詳細(xì)的介紹了解決方法,感興趣的可以了解一下2020-08-08RabbitMQ交換機(jī)與Springboot整合的簡(jiǎn)單實(shí)現(xiàn)
這篇文章主要介紹了RabbitMQ交換機(jī)與Springboot整合的簡(jiǎn)單實(shí)現(xiàn),本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-07-07