1. 程式人生 > >Java web每天學之Servlet工作原理詳情解析

Java web每天學之Servlet工作原理詳情解析

比較 access context 都是 tty 問題: character validate code

上篇文章中我們介紹了Servlet的實現方式以及Servlet的生命周期,我們這篇文章就來介紹一下常用對象。

點擊回顧:《Java Web每天學之Servlet的工作原理解析》;《Java Web每天學之Servlet的工作原理解析(二)》

一、HttpServletRequest對象

1、介紹
HttpServletRequest對象:主要作用是用來接收客戶端發送過來的請求信息,例如:請求的參數,發送的頭信息等都屬於客戶端發來的信息,service()方法中形參接收的是HttpServletRequest接口的實例化對象,表示該對象主要應用在HTTP協議上,該對象是由Tomcat封裝好傳遞過來。

HttpServletRequest是ServletRequest的子接口,ServletRequest只有一個子接口,就是HttpServletRequest。既然只有一個子接口為什麽不將兩個接口合並為一個?

從長遠上講:現在主要用的協議是HTTP協議,但以後可能出現更多新的協議。若以後想要支持這種新協議,只需要直接繼承ServletRequest接口就行了。

在HttpServletRequest接口中,定義的方法很多,但都是圍繞接收客戶端參數的。
但是怎麽拿到該對象呢?不需要,直接在Service方法中由容器傳入過來,而我們需要做的就是取出對象中的數據,進行分析、處理。

2、常用的方法
請求方式+url+版本號
getRequestURL:返回客戶端發出請求時的完整URL
getRequestURI:返回請求行中的資源名部分(項目名稱開始)
getQueryString :返回請求行中的參數部分
getMethod:得到客戶端請求方式
getProtocol:獲取HTTP版本號
getContextPath:獲取webapp名字

獲取客戶端若幹消息頭
getHeader (String name):獲取單個請求頭內容
Enumeration<String> getHeaderNames():獲取所有的請求頭名稱集合

獲得客戶機請求參數(客戶端提交的數據)
getParameter(name):獲取指定名稱的參數值,這是最為常用的方法之一。
getParameterValues(String name):獲取指定名稱參數的所有值數組。它適用於一個參數名對應多個值的情況:如頁面表單中的復選框,多選列表提交的值。

getParameterNames():返回一個包含請求消息中的所有參數名的Enumeration對象。通過遍歷這個Enumeration對象,就可以獲取請求消息中所有的參數名。

getParameterMap():返回一個保存了請求消息中的所有參數名和值的Map對象。Map對象的key是字符串類型的參數名,value是這個參數所對應的Object類型的值數組

3、請求亂碼解決
由於現在的request屬於接收客戶端的參數,所以必然有其默認的語言編碼,主要是由於在解析過程中默認使用的編碼方式為ISO-8859-1(此編碼不支持中文),所以解析時一定會出現亂碼。要想解決這種亂碼問題,需要設置request中的編碼方式,告訴服務器以何種方式來解析數據。

①req.setCharacterEncoding("UTF-8");這種方式只針對POST有效
②new String(req.getParameter(name).getBytes(“ISO-8859-1”));
Tomcat8之後GET不會出現亂碼
Tomcat8及以上 Get 不會亂碼,不用管
技術分享圖片

4、跳轉
request.getRequestDispacther(“要跳轉的地址”).forword(request,response);
請求轉發,服務器跳轉,地址欄的URL不變。

在寫路徑時,以/開頭或者 http://都是絕對路徑,不以/開頭就是相對路徑,在請求轉發時,”/”代表服務器路徑+站點名,即http://localhost:8080/站點名

請求轉發的過程:
客戶首先發送一個請求到服務器端,服務器端發現匹配的servlet,並指定它去執行,當這個servlet執行完之後,它要調用getRequestDispacther()方法,把請求轉發給指定的test.jsp,整個流程都是在服務器端完成的,而且是在同一個請求裏面完成的,因此servlet和jsp共享的是同一個request,在servlet裏面放的所有東西,在jsp中都能取出來,因此,jsp能把結果getAttribute()出來,getAttribute()出來後執行完把結果返回給客戶端。整個過程是一個請求,一個響應。

5、request域對象
通過該對象可以在一個請求中傳遞數據,作用範圍:在一次請求中有效,即服務器跳轉有效。
request.setAttribute():設置request域對象內容
request.getAttribute(String name):獲取指定request域對象內容
request.removeAttribute(String name):刪除指定request域對象

二、HttpServletResponse對象

1、介紹
Web服務器收到客戶端的http請求,會針對每一次請求,分別創建一個用於代表請求的request對象和代表響應的response對象。

request和response對象既然代表請求和響應:獲取客戶端數據,通過request對象;向客戶端輸出數據,通過response對象。

HttpServletResponse的主要功能用於服務器對客戶端的請求進行響應,將WEB服務器處理後的結果返回給客戶端。service()方法中形參接收的是HttpServletResponse接口的實例化對象,這個對象中封裝了向客戶端發送數據、發送響應頭,發送響應狀態碼的方法。

2、常用方法
addCookie(Cookie cookie):將指定的Cookie加入到當前的響應中
addHeader(String name,String value):將指定的鍵值加入到響應頭信息中
containsHeader(String name) :返回一個布爾值,判斷響應的頭部是否被設置
encodeURL(String url):編碼指定的URL
sendError(int sc):使用指定狀態碼發送一個錯誤到客戶端
sendRedirect(String location):發送一個臨時響應到客戶端,重定向
setHeader(String name,String value):將給出的名字和值設置響應的頭部
setStatus(int sc):給當前響應設置狀態
setContentType(String ContentType):設置響應的MIME類型
getWriter():獲取輸出字符流
getOutputStream():獲取輸出的字節流

3、設置頭信息
1)、刷新、跳轉頁面
所有頭信息都是隨著請求和回應自動發送到服務器端(客戶端),在response中一個比較常用的頭信息就是刷新的指令,可以完成定時刷新的功能。
resp.setHeader("refresh","2");

對於刷新的頭信息,除了定時的功能外,還具備了定時跳轉的功能,可以讓一個頁面定時跳轉到一個指定的頁面。(已經註冊成功,兩秒後跳轉到登陸頁面)
response.setHeader("refresh","3;URL=ok.html");

但是這種跳轉不是萬能的,有時候根本就無法進行跳轉操作,返回後刷新不會跳轉;
對於這種定時跳轉的頭信息,也可以采用HTML的方式進行設置,HTML本身也可以設置頭信息。(客戶端跳轉)
<meta http-equiv="refresh" content="3;http://www.shsxt.com>

2)、返回狀態碼
setStatus方法用來設置Servlet向客戶端返回的狀態碼,它用來設置沒有出錯的狀態,通常不使用。如果Servlet運行出錯,Servlet可以使用sendError方法設置狀態碼,如sendError(int sc)方法設置錯誤狀態代碼。sendError(int sc,String msg)方法除了設置狀態碼,還向客戶發出一條錯誤信息。
3)、添加Cookie
addCookie方法可以在Web服務器響應中加入Cookie對象,這個對象將被瀏覽器所保存。Cookie機制也被用來維護會話狀態。
4、跳轉
response.sendRedirect(“要跳轉的路徑”);
通過response也可以進行跳轉,該種跳轉稱為請求重定向,屬於客戶端跳轉,地址欄改變,存在兩個請求。請求中的參數不能在此種跳轉間傳遞。
在寫路徑時,以/開頭或者http://都是絕對路徑,不以/開頭就是相對路徑,在重定向時,”/”代表服務器路徑,即http://localhost:8080

5、重定向過程:
客戶瀏覽器發送http請求--->web服務器接受後發送302狀態碼響應及對應新的location給客戶瀏覽器-->瀏覽器發現是302響應,自動再發送一個新的http請求,請求的url是新的location地址-->服務器根據此請求尋找資源並發送給客戶。

6、響應亂碼解決
當服務器端響應給客戶端中文時,可能存在亂碼
兩種方式響應回數據:PrintWriter 和 ServletOutputStream
getOutputStream和getWriter這兩個方法互相排斥,調用了其中的任何一個方法後,就不能再調用另一方法。
PrintWriter():一定亂碼,服務器自動使用ISO-8859-1編碼,通過如下方式解決亂碼
response.setCharacterEncoding("utf-8");
response.setHeader("content-type", "text/html;charset=utf-8");
一句頂兩句:
response.setContentType("text/html;charset=utf-8");
OutputStream() :可能亂碼

7、使用OutputStream流輸出中文註意問題:
服務器端,數據輸出的編碼方式,就是客戶端打開的編碼方式
例:outputStream.write("中國".getBytes("utf-8"));使用OutputStream流向瀏覽器輸出中文,編碼方式是utf-8;此時客戶端瀏覽器也要以utf-8的編碼打開,否則會出現中文亂碼。在服務器端控制客戶端瀏覽器以utf-8的編碼方式顯示,此時既可以通過上述解決PrintWriter 的方式,也可以通過設置響應頭信息的方式:
response.setHeader ("content-type", "text/html;charset=UTF-8");

三、HttpSession對象
1、介紹
HttpSession對象是javax.servlet.http.HttpSession的實例,該接口並不像HttpServletRequest或HttpServletResponse還存在一個父接口,該接口只是一個純粹的接口。因為session屬於HTTP協議的範疇。

對於服務器而言,每一個連接到它的客戶端都是一個session,servlet 容器使用此接口創建 HTTP 客戶端和 HTTP 服務器之間的會話。會話將保留指定的時間段,跨多個連接或來自用戶的頁面請求。一個會話通常對應於一個用戶,該用戶可能多次訪問一個站點。可以通過此接口查看和操作有關某個會話的信息,比如會話標識符、創建時間和最後一次訪問時間。在整個session中,最重要的就是屬性的操作
session無論客戶端還是服務器端都可以取得,若重新打開一個新的瀏覽器,則無法取得之前設置的session,因為每一個session只保存在當前的瀏覽器當中,並在相關的頁面取得。

2、使用
常用方法
setAttribute(String name,Object value):將value對象以name名稱綁定到會話
getAttribute(String name):取得name的屬性值,如果屬性不存在則返回null
removeAttribute(String name):從會話中刪除name屬性,若不存在不會執行,也不會拋處錯誤。
Enumeration getAttributeNames():返回和會話有關的枚舉值
invalidate():使會話失效,同時刪除屬性對象
isNew():用於檢測當前客戶是否為新的會話
long getCreationTime():返回會話創建時間
getLastAccessedTime():返回會話時間內web容器接收客戶最後發出的請求時間
int getMaxInactiveInterval():返回在會話期間內客戶請求的最長時間為秒
setMaxInactiveInterval(int seconds):允許客戶客戶請求的最長時間
ServletContext getServletContext():返回當前會話的上下文環境,ServletContext對象可以使Servlet與web容器進行通信
String getId():返回會話期間的識別號

Session:默認是在第一次使用Session時被創建
Cookie: 大部分都是通過服務器主動(程序手動)設置到瀏覽器的, 只有JSESSIONID是服務自己設定的, 不需要程序員操心
session id

對於每一個用戶而言,實際上都表示一個個不同的session,服務器通過session id來區分這些用戶,即每一個通過瀏覽器連接到服務器上的用戶都會由服務器分配一個唯一的不會重復的編號。
cookie自動設置的JSESSIONID的值就是每一個用戶的session id,所以session在進行操作的時候用到了cookie的處理機制。

session id的結束
① 生命周期到期(默認一次瀏覽器的打開與關閉)
②非正常關閉服務器session銷毀
正常關閉服務器,session對象不會被銷毀,而是會被序列化到磁盤上
工作空間work目錄下 SESSION.ser
③ 調用session的 invalidate() 方法
session在所有的項目開發中,用得最多的地方就是登陸驗證及註銷操作功能
3、session域對象
一個會話創建一個HttpSession對象,同一會話中的多個請求中可以共
享session中的數據。
request.getSession().setAttribute():設置session域對象內容
request.getSession().getAttribute(String name):獲取指定session域對象內容
request.getSession().removeAttribute(String name):刪除指定session域對象

四、ServletContext對象
1、介紹
一個web應用只有一個ServletContext對象,所有的servlet都共享這個ServletContext對象,又稱為Application對象。WEB容器在啟動時,會為每個WEB應用程序創建一個對應的ServletContext對象,其代表當前WEB應用。ServletContext定義了一組方法,servlet 使用這些方法與其他 servlet 容器進行通信,ServletContext 對象包含在 ServletConfig 對象中,ServletConfig對象在初始化 servlet 時由 web 服務器提供給 servlet。

2、常用方法
getInitParameter(String name):獲取初始化參數
getResource(String parh)方法:其中path必須是/開頭,代表當前web應用程序的根目錄。返回返回的一個代表某個資源的URL對象。
getResoutceAsStream(String parh):返回文件流。這個好處是可以使用相對於根目錄的路徑訪問到web目錄下的所有文件,而不必知道絕對路徑。
getContextPath():得到web應用的路徑
getRealPath():獲取絕對路徑

3、servletcontext域對象
ServletContext是一個全局的儲存信息的空間,服務器開始就存在,服務器關閉才釋放。
request,一個用戶可有多個;session,一個用戶一個;而servletContext,所有用戶共用一個。所以,為了節省空間,提高效率,ServletContext中,要放必須的、重要的、所有用戶需要共享的線程又是安全的一些信息。
request.getServletContext().setAttribute():設置域對象內容
request.getServletContext().getAttribute(String name):獲取域對象內容
request.getServletContext().removeAttribute(String name):刪除域對象

上海Java培訓感謝您的閱讀,轉載請註明出處,請多多關註支持,陸續Java相關技術文章奉上!

Java web每天學之Servlet工作原理詳情解析