一、Servlet概述
1.sun公司提供的動態(tài)web資源開發(fā)技術(shù)。本質(zhì)是上一段java小程序,要求這個小程序必須實現(xiàn)Servlet接口,以便服務(wù)器能夠調(diào)用。
2.開發(fā)Servlet的兩個步驟*實驗:
Servlet的快速入門
(1)步驟一:寫一個java程序?qū)崿F(xiàn)Servlet接口(此處直接繼承了默認(rèn)實現(xiàn)類GenericServlet)
package cn.itheima; import java.io.*;
import javax.servlet.*;
public class FirstServlet extends GenericServlet{
????public void service(ServletRequest req, ServletResponse res) throws ServletException, java.io.IOException{ ????????res.getOutputStream().write("My FirstServlet!".getBytes());
????}
}
(2)將編譯好的帶包的.class放到WEB-INF/classes下以外,還要配置web應(yīng)用的 web.xml注冊

二、Servlet的詳述
1.生命周期:一件事物什么時候生,什么時候死,在生存期間必然會做的事情,所有這些放在一起就是該事物的聲明周期。
2.Servlet的生命周期:通常情況下,servlet第一次被訪問的時候在內(nèi)存中創(chuàng)建對象,在創(chuàng)建后立即調(diào)用init()方法進行初始化。對于每一次請求都掉用service(req,resp)方法處理請求,此時會用Request對象封裝請求信息,并用Response對象(最初是空的)代表響應(yīng)消息,傳入到service方法里供使用。當(dāng)service方法處理完成后,返回服務(wù)器服務(wù)器根據(jù)Response中的信息組織稱響應(yīng)消息返回給瀏覽器。響應(yīng)結(jié)束后servlet并不銷毀,一直駐留在內(nèi)存中等待下一次請求。直到服務(wù)器關(guān)閉或web應(yīng)用被移除出虛擬主機,servlet對象銷毀并在銷毀前調(diào)用destroy()方法做一些善后的事情。
3.Servlet接口的繼承結(jié)構(gòu)
Servlet接口:定義了一個servlet應(yīng)該具有的方法,所有的Servlet都應(yīng)該直接或間接實現(xiàn)此接口
????|
????|----GenericServlet:對Servlet接口的默認(rèn)實現(xiàn),通用Servlet,這是一個抽象類,其中的大部分方法都做了默認(rèn)實現(xiàn),只有service方法是一抽????象方法需要繼承者自己實現(xiàn)
????????????????????|
? ? ? ? ? ? ? ? ? ? |----HttpServlet:對HTTP協(xié)議進行了優(yōu)化的Servlet,繼承自GenericServlet類,并且實現(xiàn)了其中的service抽象方法,默認(rèn)的實現(xiàn)中判斷了請求的請求方式,并根據(jù)請求方式的不同分別調(diào)用不同的doXXX()方法。通常我們直接繼承HttpServlet即可
4.6servlet的線程安全問題
4.6.1由于通常情況下,一個servlet在內(nèi)存只有一個實例處理請求,當(dāng)多個請求發(fā)送過來的時候就會有多個線程操作該servlet對象,此時可能導(dǎo)致線程安全問題。
(1)serlvet的成員變量可能存在線程安全問題
*實驗:定義一個成員變量 int i = 0;在doXXX()方法中進行i++操作并輸出i值到客戶端,此時由于延遲可能導(dǎo)致線程安全問題
(2)serlvet操作資源文件時,多個線程操作同一文件引發(fā)線程安全問題
*實驗:請求帶著一個參數(shù)過來,servlet將請求參數(shù)寫入到一個文件,再讀取該文件,將讀取到的值打印到客戶端上,有可能有線程安全問題
4.6.2解決方法
(1)利用同步代碼塊解決問題。缺陷是,同一時間同步代碼塊只能處理一個 請求,效率很低下,所以同步代碼塊中盡量只包含核心的導(dǎo)致線程安全問題的代碼。
(2)為該servlet實現(xiàn)SingleThreadModel接口,此為一個標(biāo)記接口,被標(biāo)記的servlet將會在內(nèi)存中保存一個servlet池,如果一個線程來了而池中沒有servlet對象處理,則創(chuàng)建一個新的。如果池中有空閑的servlet則直接使用。這并不能真的解決線程安全問題。此接口已經(jīng)被廢棄。
(3)兩種解決方案都不夠完美,所以盡量不要在servlet中出現(xiàn)成員變量。
三、ServletConfig
????????????1.代表servlet配置的對象,可以在web.xml中<servlet>中配置
?? ? ? <servlet>
????????????<servlet-name>Demo5Servlet</servlet-name>
<servlet-class>cn.it.Demo5SErvlet</servlet-class>
? ? <init-param>
? ? ? ? <param-name>data1</param-name>
? ? ? ? <param-value>value1</param-value>
? ? </init-param>
</servlet>
然后在servlet中利用this.getServletConfig()獲取ServletConfig對象,該對象提供了getInitParameter()和getInitParameterNames()方法,可以遍歷出配置中的配置項。 不想在servlet中寫死的內(nèi)容可以配置到此處。
四、ServletContext
1.代表當(dāng)前web應(yīng)用的對象。
2.作為域?qū)ο笫褂?,在不同servlet之間傳遞數(shù)據(jù),作用范圍是整個web應(yīng)用生命周期:當(dāng)web應(yīng)用被加載進容器時創(chuàng)建代表整個web應(yīng)用的ServletContext對象。
當(dāng)服務(wù)器關(guān)閉或web應(yīng)用被移除出容器時,ServletContext對象跟著銷毀。~域:一個域就理解為一個框,這里面可以放置數(shù)據(jù),一個域既然稱作域,他就有一個可以被看見的范圍,這個范圍內(nèi)都可以對這個域中的數(shù)據(jù)進行操作,那這樣的對象就叫做域?qū)ο蟆?/p>
3.在web.xml可以配置整個web應(yīng)用的初始化參數(shù),利用ServletContext去獲得param1pvalue1this.getServletContext().getInitParameter("param1")this.getServletContext().getInitParameterNames()
4.在不同servlet之間進行轉(zhuǎn)發(fā)this.getServletContext().getRequestDispatcher("/servlet/Demo10Servlet").forward(request, response);方法執(zhí)行結(jié)束,service就會返回到服務(wù)器,再有服務(wù)器去調(diào)用目標(biāo)servlet,其中request會重新創(chuàng)建,并將之前的request的數(shù)據(jù)拷貝進去。
5.讀取資源文件5.1由于相對路徑默認(rèn)相對的是java虛擬機啟動的目錄,所以我們直接寫相對路徑將會是相對于tomcat/bin目錄,所以是拿不到資源的。如果寫成絕對路徑,當(dāng)項目發(fā)布到其他環(huán)境時,絕對路徑就錯了。
5.2為了解決這個問題ServletContext提供了this.getServletContext().getRealPath("/1.properties"),給進一個資源的虛擬路徑,將會返回該資源在當(dāng)前環(huán)境下的真實路徑。this.getServletContext().getResourceAsStream("/1.properties"),給一個資源的虛擬路徑返回到該資源真實路徑的流。5.3當(dāng)在非servlet下獲取資源文件時,就沒有ServletContext對象用了,此時只能用類加載器classLoader.getResourceAsStream("../../1.properties"),此方法利用類加載器直接將資源加載到內(nèi)存中,有更新延遲的問題,以及如果文件太大,占用內(nèi)存過大。classLoader.getResource("../1.properties").getPath(),直接返回資源的真實路徑,沒有更新延遲的問題。