一、Http
1、簡介
(超文本傳輸協議)是一個簡單的請求-回應協議,它通常運行在TCP之上,
-
文本:html,字串,…
-
超文本:圖片,音樂,視頻,定位,地圖.……
-
埠:80
2、Http請求
客戶端——>發請求(Request)——>服務器
3、Http回應
服務器——>回應——>客戶端
二、Servlet
1、簡介
Servlet是運行在服務器上的一個java小程式,它可以接收客戶端發送過來的請求,并回應資料給客戶端
開發一個Servlet程式需要完成兩個步驟:
-
撰寫一個類,實作Serlet介面
-
把開發好java類部署到web服務器中
Servlet三大組件:
-
Servlet程式
-
Filter過濾器
-
Listener監聽器
2、Servlet缺點
-
一個Servlet只能完成一個業務處理
-
每個Servlet的配置資訊至少8行
-
利用HttpServletRequest是獲取客戶端引數唯一方法, 并且獲取的引數型別都是String, 可能需要做轉換處理
-
Servlet處理用戶的請求型別比較單一(doGet/doPost)
3、HelloWorldServlet
構建一個普通的Maven Web,專案,選擇模板時勾選maven-archetype-webapp,洗掉src目錄,pom.xml配置如下:

構建新module:HelloWorldServlet
撰寫Servlet程式,繼承Servlet介面的實作類javax.servlet.GenericServlet來實作

修改HelloWorldServlet.web.xml

撰寫Servlet的映射

配置Tomcat服務器,運行程式

4、Servlet作業原理
部署Web應用服務,利用Tomcat啟動應用程式,此時Tomcat作為Servlet容器,會創建并初始化Servlet;

通過瀏覽器訪問Web應用服務器,利用Http通訊協議傳遞資料;

當訪問到達時,Servlet容器創建HttpServletRequest和HttpServletResponse物件,并且將請求資料放入HttpServletRequest中;

Servlet容器根據傳遞的url通過匹配web.xml中配置的Servlet路徑,找到對應的Servlet,傳遞請求和回應物件;

Servlet中根據請求方式呼叫doGet或doPost方法,處理請求資料,將回應資料放入HttpServletResponse,回傳給Servlet容器;

Servlet容器決議回應資料,根據HTTP通訊協議,回傳給瀏覽器,瀏覽器決議并展示資料;

5、Servlet作業流程

-
瀏覽器向服務器發出GET請求(請求服務器ServletA)
-
服務器上的容器邏輯接收到該url,根據該url判斷為Servlet請求,此時容器邏輯將產生兩個物件:請求物件(HttpServletRequest)和回應物件(HttpServletResponce)
-
容器邏輯根據url找到目標Servlet(本示例目標Servlet為ServletA),且創建一個執行緒A
-
容器邏輯將剛才創建的請求物件和回應物件傳遞給執行緒A
-
容器邏輯呼叫Servlet的service()方法
-
service()方法根據請求型別(本示例為GET請求)呼叫doGet()(本示例呼叫doGet())或doPost()方法
-
doGet()執行完后,將結果回傳給容器邏輯
-
執行緒A被銷毀或被放在執行緒池中
注意:
-
在容器中的每個Servlet原則上只有一個實體
-
每個請求對應一個執行緒
-
多個執行緒可作用于同一個Servlet(這是造成Servlet執行緒不安全的根本原因)
-
每個執行緒一旦執行完任務,就被銷毀或放在執行緒池中等待回收
6、Servlet生命周期

其中:
-
init() 方法:init() 方法是在創建 Servlet 物件時被呼叫,而且只能被呼叫一次,用于 Servlet 物件在整個生命周期內的唯一一次初始化,只有在 init() 方法呼叫成功后,Servlet 才會處于服務狀態,才能夠去處理客戶端的請求,
-
service() 方法:service() 方法是 Servlet 作業的核心方法,當客戶端請求訪問 Servlet 時,Servlet 容器就會呼叫 service() 方法去處理來自客戶端的請求,并把處理后的回應回傳給客戶端,
-
destroy() 方法:destory() 方法是 Servlet 容器回收 Servlet 物件之前呼叫的,且只會呼叫一次,而此時的服務器處于停止狀態或者訪問資源已經被移除,
(1)、初始化階段
當用戶第一次向 Servlet 容器發出 HTTP 請求要求訪問某個 Servlet 時,Servlet 容器會在整個容器中搜索該 Servlet 物件,發現這個 Servlet 物件沒有被實體化,于是創建這個 Servlet 物件,然后呼叫該物件的 init() 方法完成初始化, 當用戶第二次訪問這個 Servlet 時,Servlet 容器仍然在容器中搜索該 Servlet 物件,結果找到了該物件的實體,則不去創建而直接使用該物件, 找到了對應的 Servlet 物件,隨后 Servlet 進入到運行階段, 需要注意的是,在 Servlet 的整個生命周期內,它的 init() 方法只被呼叫一次,
(2)、運行階段
這是 Servlet 生命周期中最核心的階段,在該階段中,Servlet 容器會為當前的請求創建一個 ServletRequest 物件和一個 ServletResponse 物件(它們分別代表 HTTP 請求和 HTTP 回應),并將這兩個物件作為引數傳遞給 Servlet 的 service() 方法, service() 方法從 ServletRequest 物件中獲得用戶的詳細請求資訊并處理該請求,通過 ServletResponse 物件生成回應結果, 需要強調的是,在 Servlet 的整個生命周期內,用戶每次請求訪問 Servlet 時,Servlet 容器都會呼叫一次 Servlet 的 service() 方法,并且創建新的 ServletRequest 和 ServletResponse 物件,
(3)、銷毀階段
當服務器停止時,Servlet 容器需要回收 Servlet 物件所占用的記憶體,在回收之前,會自動呼叫該物件的 destroy() 方法做好回收記憶體前的準備,辟如關閉后臺執行緒, 和 init() 方法類似,destroy() 方法也只會被呼叫一次,
注意:Servlet 物件一旦創建就會駐留在記憶體中一直等待客戶端的訪問,直到服務器關倍訓專案被移除出容器時,Servlet 物件才會被銷毀,
總結:
-
Servlet 物件被創建之后執行 init() 方法,并且 init() 方法只執行一次,其主要目的是完成 Servlet 物件的初始化作業,
-
對于 Servlet 物件的 service() 方法,只要用戶請求一次,那么 service() 方法就執行一次,
-
Servlet 物件被回收之前,destroy() 方法會被呼叫,該方法只執行一次,執行的時候 Servlet 物件還在,
三、Servlet高級應用
1、HttpServlet
(1)、簡介
HttpServlet是Servlet介面的一個實作類,并且它是一個抽象類,servlet.http包中定義了采用HTTP通信協議(一個無狀態協議)的HttpServlet類,HttpServlet繼承了GenericServlet,而GenericServlet實作Servlet介面,所以我們可以同構繼承HttpServlet來創建一個Servlet,
(2)、回應流程
-
Web客戶向Servlet容器發出Http請求
-
Servlet容器決議Web客戶的Http請求
-
Servlet容器創建一個HttpRequest物件,在這個物件中封裝Http請求資訊
-
Servlet容器創建一個HttpResponse物件
-
Servlet容器呼叫HttpServlet的service方法,把HttpRequest和HttpResponse物件作為service方法的引數傳給HttpServlet物件
-
HttpServlet呼叫HttpRequest的有關方法,獲取HTTP請求資訊
-
HttpServlet呼叫HttpResponse的有關方法,生成回應資料
-
Servlet容器把HttpServlet的回應結果傳給Web客戶
其中HttpServlet首先必須讀取Http請求的內容,Servlet容器負責創建HttpServlet物件,并把Http請求直接封裝到HttpServlet物件中,
(3)、創建HttpServlet步驟
-
繼承HttpServlet抽象類
-
重寫HttpServlet的部分方法,如doGet()或doPost()方法
-
獲取HTTP請求資訊,通過HttpServletRequest物件來檢索HTML表單所提交的資料或URL上的查詢字串
-
生成HTTP回應結果,通過HttpServletResponse物件生成回應結果
(4)、具體實作
創建HttpServlet_HelloWorld具體實作類

配置xml

配置Tomcat,運行程式

2、Servlet請求映射
在 web.xml 檔案中,一個 <servlert-mapping> 元素用于映射一個Servlet
(1)、Servlet的多重映射
Servlet 的多重映射指同一個 Servlet 可以被映射成多條虛擬路徑,也就是說,客戶端可以通過多條路徑實作對同一個 Servlet 的訪問,
a、配置多個 <servlet-mapping> 元素

啟動Tomcat

b、配置多個 <url-pattern> 子元素
在 web.xml 檔案中對 TestServlet01 的 <servlet-mapping> 元素進行修改,在該元素中配置多個 <url-pattern>

啟動Tomcat

(2)、使用通配符
在實際開發程序中,開發者有時會希望某個目錄下的所有路徑都可以訪問同一個 Servlet,這時,可以在 Servlet 映射的路徑中使用通配符*,通配符的格式有兩種,具體如下,
-
格式為“*.擴展名”,例如 *.do 匹配以 .do 結尾的所有 URL 地址,
-
格式為 /*,例如 /abc/* 匹配以 /abc 開始的所有 URL 地址,
需要注意的是,這兩種通配符的格式不能混合使用,例如,/abc/*.do 是不合法的映射路徑,另外,當客戶端訪問一個 Servlet 時,如果請求的 URL 地址能夠匹配多條虛擬路徑,那么 Tomcat 將采取最具體匹配原則查找與請求 URL 最接近的虛擬映射路徑,例如,對于如下所示的一些映射關系:
/abc/* 映射到 Servlet1 /* 映射到 Servlet2 /abc 映射到 Servlet3 *.do 映射到 Servlet4
-
當請求 URL 為 /abc/a.html,/abc/* 和 /* 都可以匹配這個 URL,Tomcat 會呼叫 Servlet1,
-
當請求 URL 為 /abc、/*、/abc/* 和 /abc 都可以匹配這個 URL,Tomcat 會呼叫 Servlet3,
-
當請求 URL 為 /abc/a.do、/*、*.do 和 /abc/* 都可以匹配這個 URL,Tomcat 會呼叫 Servlet1,
-
當請求 URL 為 /a.do,/* 和 *.do 都可以匹配這個 URL,Tomcat 會呼叫 Servlet2,
-
當請求 URL 為 /xxx/yyy/a.do,*.do 和 /* 都可以匹配這個 URL,Tomcat 會呼叫 Servlet2,
(3)、默認servlet
如果某個 Servlet 的映射路徑僅僅是一個正斜線(/),那么這個 Servlet 就是當前 Web 應用的默認 Servlet,Servlet 服務器在接收到訪問請求時,如果在 web.xml 檔案中找不到匹配的 <servlet-mapping> 元素的 URL,則會將訪問請求交給默認 Servlet 處理,也就是說,默認 Servlet 用于處理其他 Servlet 都不處理的訪問請求,

當 URL 地址和 TestServlet01 的虛擬路徑不匹配時,瀏覽器仍然可以正常訪問HelloWorldServlet ,

3、ServletConfig
代表當前Servlet在web.xml中的配置資訊(用的不多)
-
String getServletName() -- 獲取當前Servlet在web.xml中配置的名字
-
String getInitParameter(String name) -- 獲取當前Servlet指定名稱的初始化引數的值
-
Enumeration getInitParameterNames() -- 獲取當前Servlet所有初始化引數的名字組成的列舉
-
ServletContext getServletContext() -- 獲取代表當前web應用的ServletContext物件
在Servlet的組態檔中,可以使用一個或多個<init-param>標簽為servlet配置一些初始化引數,
當servlet配置了初始化引數后,web容器在創建servlet實體物件時,會自動將這些初始化引數封裝到ServletConfig物件中,并在呼叫servlet的init方法時,將ServletConfig物件傳遞給servlet,進而,程式員通過ServletConfig物件就可以得到當前servlet的初始化引數資訊,
這樣做的好處是:如果將資料庫資訊、編碼方式等配置資訊放在web.xml中,如果以后資料庫的用戶名、密碼改變了,則直接很方便地修改web.xml就行了,避免了直接修改源代碼的麻煩,
新建一個名為ServletConfigTest的Servlet,然后在web.xml中的<servlet>標簽下,通過<init-param>標簽為這個servlet配置兩個初始化引數:

然后在代碼中獲取上面的兩個引數,代碼實作如下:

運行程式,后臺列印日志如下:

4、ServletContext
代表當前web應用(非常重要)
WEB容器在啟動時,它會為每個WEB應用程式都創建一個對應的ServletContext物件,它代表當前web應用,
ServletConfig物件中維護了ServletContext物件的參考,開發人員在撰寫servlet時,可以通過ServletConfig.getServletContext方法獲得ServletContext物件,
由于一個WEB應用中的所有Servlet共享同一個ServletContext物件,因此Servlet物件之間可以通過ServletContext物件來實作通訊,ServletContext物件通常也被稱之為context域物件,
應用:
(1)、共享資料
-
域物件:在一個可以被看見的范圍內共享資料用到物件
-
作用范圍:整個web應用范圍內共享資料
-
生命周期:當服務器啟動web應用加載后創建出ServletContext物件后,域產生,當web應用被移除出容器或服務器關閉,隨著web應用的銷毀域銷毀,
servletcontext01.java

servletcontext02.java

我們在servletcontext0中給Context加一個引數name(16行),然后就能在servletcontext02中得到這個引數了(16行),
context中常用的方法有:
-
void setAttribute(String,Object);
-
Object getAttribute(String);
-
void removeAttribute(String);
(2)、獲取初始化引數
我們在第一段中,通過<init-param>標簽為某一個單獨的servlet加配置資訊,這種配置資訊在其他的Servlet中是無法訪問到的,可如果我們使用<context-param>標簽(與Servlet標簽并列)為整個Web應用配置屬性
在web.xml中為整個web應用添加初始化引數:用戶名、密碼,代碼位置如下:

servlet注冊

ServletContext03.java

運行程式

(3)、請求轉發
-
請求重定向:302+Location(兩次請求兩次回應)
-
請求轉發:服務器內不進行資源流轉 (一次請求一次回應,來實作資源流轉)
注:上方括號中的內容是二者的區別,打個比方,假如你找我借錢,如果是請求重定向的話,那你再去找別人借;如果是請求轉發的話,那我去找別人借,然后再借給你,

ServletContext04.java

通過19行代碼拿到轉發器dispatcher,這個轉發器指向ServletTest05(引數中寫虛擬路徑),然后一旦執行完20行代碼,就會跳到ServletTest05中去執行,
ServletContext05.java

web.xml

運行

轉載請註明出處,本文鏈接:https://www.uj5u.com/qita/382151.html
標籤:其他
